当前位置: 首页 > news >正文

React 组件的通信方式

在 React 应用开发中,组件之间的通信是构建复杂用户界面和交互逻辑的关键。正确地实现组件通信能够让我们的应用更加灵活和易于维护。以下是几种常见的 React组件通信方式。

一、父子组件通信

1. 通过 props 传递数据(父组件向子组件传递数据)

  • 原理:
    在 React 中,props 是一种从父组件向子组件传递数据的机制。父组件在渲染子组件时,可以将数据作为属性传递给子组件。子组件通过接收 props 来获取父组件传递的数据,并在自身的渲染过程中使用这些数据。这种方式遵循了单向数据流的原则,即数据从父组件流向子组件,使得数据的流动清晰可追踪。
  • 示例代码:
// 父组件
import React from 'react';
import ChildComponent from './ChildComponent';function ParentComponent() {const userName = 'John Doe';const userAge = 30;return (<div><ChildComponent name={userName} age={userAge} /></div>);
}export default ParentComponent;// 子组件
import React from 'react';function ChildComponent(props) {return (<div><p>Name: {props.name}</p><p>Age: {props.age}</p></div>);
}export default ChildComponent;

2. 父组件通过回调函数接收子组件传递的数据(子组件向父组件通信)

  • 原理:
    当子组件需要向父组件传递数据时,父组件可以向子组件传递一个回调函数作为 props。子组件在特定的事件触发时(如按钮点击、表单提交等)调用这个回调函数,并将需要传递的数据作为参数传递给该回调函数。这样父组件就能接收到子组件传来的数据,实现反向的数据流动。
  • 示例代码:
// 父组件
import React from 'react';
import ChildComponent from './ChildComponent';function ParentComponent() {const handleChildMessage = (message) => {console.log('Received message from child:', message);};return (<div><ChildComponent onMessage={handleChildMessage} /></div>);
}export default ParentComponent;// 子组件
import React from 'react';function ChildComponent(props) {const sendMessage = () => {const message = 'Hello from child!';props.onMessage(message);};return (<button onClick={sendMessage}>Send Message</button>);
}export default ChildComponent;

二、兄弟组件通信

1. 利用共同父组件实现通信

  • 通过共同父组件中转数据:
    当两个兄弟组件需要通信时,可以通过它们的共同父组件来实现。一个兄弟组件通过父组件传递的回调函数向父组件传递数据,父组件再将数据通过 props 传递给另一个兄弟组件。本质是父子之间通信
  • 示例代码:
// 父组件
import React from 'react';
import SiblingA from './SiblingA';
import SiblingB from './SiblingB';function ParentComponent() {const [message, setMessage] = React.useState('');const handleMessageFromA = (msg) => {setMessage(msg);};return (<div><SiblingA onSendMessage={handleMessageFromA} /><SiblingB message={message} /></div>);
}export default ParentComponent;// 兄弟组件 A
import React from 'react';function SiblingA(props) {const sendMessage = () => {const msg = 'Hello from Sibling A';props.onSendMessage(msg);};return (<button onClick={sendMessage}>Send Message to Sibling B</button>);
}export default SiblingA;// 兄弟组件 B
import React from 'react';function SiblingB(props) {return (<div>{props.message}</div>);
}export default SiblingB;

三、跨级组件通信

1. 使用 React Context API

  • 原理:
    React Context 提供了一种在组件树中跨多个层级共享数据的方法,无需通过一级一级地传递 props。它允许我们创建一个上下文(Context),在这个上下文中的组件都可以访问到共享的数据。当数据在顶层的 Context.Provider 中更新时,所有订阅了该 Context 的组件都会收到通知并重新渲染。
  • 示例代码:

Context允许我们在组件树中传递数据,而不必手动通过Props一层层传递

创建:

  • 使用 React.createContext() 创建上下文对象
  • 并在组件中使用 Provider 组件的value属性提供数据
  • 子组件通过 Consumer 或 useContext 获取数据。
// 创建 Context
import React from 'react';const ThemeContext = React.createContext();// 顶层组件(提供数据)
function App() {const theme = {color: 'blue',backgroundColor: 'lightblue'};return (<ThemeContext.Provider value={theme}><Header /></ThemeContext.Provider>);
}// 中间组件(无需关心 Context 数据)
function Header() {return (<div><Navigation /></div>);
}// 底层组件(使用 Context 数据)
function Navigation() {return (<ThemeContext.Consumer>{theme => (<ul style={theme}><li>Home</li><li>About</li><li>Contact</li></ul>)}</ThemeContext.Consumer>);
}

在函数式组件中,更常用的是 useContext 钩子来获取 Context 中的数据,示例如下:

import React, { useContext } from 'react';const ThemeContext = React.createContext();function App() {const theme = {color: 'blue',backgroundColor: 'lightblue'};return (<ThemeContext.Provider value={theme}><Header /></ThemeContext.Provider>);
}function Header() {return (<div><Navigation /></div>);
}function Navigation() {const theme = useContext(ThemeContext);return (<ul style={theme}><li>Home</li><li>About</li><li>Contact</li></ul>);
}

这样,无论组件嵌套层级有多深,都可以通过 Context API 方便地实现跨层级组件通信。

四、非父子关系组件通信(发布订阅模式)

1. 使用Event Bus进行通信

对于一些非父子关系且较为松散的组件通信场景,可以使用发布订阅模式。在 React 中,可以借助第三方库如 mitt 来实现。

首先安装 mitt 库:npm install mitt

例如,有组件 ComponentA 和 ComponentB,它们之间没有直接的父子或兄弟关系。在一个单独的事件总线文件 eventBus.js 中:

import mitt from'mitt';const emitter = mitt();export default emitter;

在 ComponentA 中,当某个事件发生时发布消息:

import React from'react';
import emitter from './eventBus';const ComponentA = () => {const handleClick = () => {const data = "来自组件 A 的消息";emitter.emit('customEvent', data);};return (<div><button onClick={handleClick}>触发事件</button></div>);
};export default ComponentA;

在 ComponentB 中,订阅该事件并接收数据:

import React, { useEffect } from'react';
import emitter from './eventBus';const ComponentB = () => {useEffect(() => {const subscription = emitter.on('customEvent', (data) => {console.log(`收到组件 A 消息: ${data}`);});return () => {// 组件卸载时取消订阅subscription();};}, []);return (<div><p>组件 B 等待接收消息</p></div>);
};export default ComponentB;

五、使用 Redux 等状态管理库

  • 原理:
    Redux 是一个可预测的状态容器,用于管理 JavaScript 应用中的状态。在 React 应用中结合 Redux,我们可以将数据存储在一个单一的 store 中。组件可以通过 dispatch 动作(action)来触发状态的改变,而通过订阅 store 的变化,组件可以获取到最新的状态。这种方式适合于大型应用中复杂的跨级组件通信场景,使得状态的管理更加集中和可维护。
  • 示例代码(简化版) :
    首先,安装 redux 和 react-redux 库。
// 定义 action 类型
const SET_USER_INFO = 'SET_USER_INFO';// action 创建函数
const setUserInfoAction = (userInfo) => ({type: SET_USER_INFO,payload: userInfo
});// reducer 函数
const initialState = {user: null
};
const rootReducer = (state = initialState, action) => {switch (action.type) {case SET_USER_INFO:return {...state, user: action.payload };default:return state;}
};// 创建 store
import { createStore } from 'redux';
const store = createStore(rootReducer);// 在顶层组件中 dispatch 动作
import React from 'react';
import { Provider } from 'react-redux';
import ChildComponent from './ChildComponent';function App() {const handleSetUserInfo = () => {const userInfo = { name: 'Alice', age: 25 };store.dispatch(setUserInfoAction(userInfo));};return (<Provider store={store}><button onClick={handleSetUserInfo}>Set User Info</button><ChildComponent /></Provider>);
}// 在子组件中获取 store 中的数据
import React from 'react';
import { useSelector } from 'react-redux';function ChildComponent() {const user = useSelector(state => state.user);return (<div>{user && (<p>User Name: {user.name}, Age: {user.age}</p>)}</div>);
}

相关文章:

React 组件的通信方式

在 React 应用开发中&#xff0c;组件之间的通信是构建复杂用户界面和交互逻辑的关键。正确地实现组件通信能够让我们的应用更加灵活和易于维护。以下是几种常见的 React组件通信方式。 一、父子组件通信 1. 通过 props 传递数据&#xff08;父组件向子组件传递数据&#xff0…...

WAV文件双轨PCM格式详细说明及C语言解析示例

WAV文件双轨PCM格式详细说明及C语言解析示例 一、WAV文件双轨PCM格式详细说明1. WAV文件基本结构2. PCM编码方式3. 双轨PCM格式详细说明二、C语言解析WAV文件的代码示例代码说明一、WAV文件双轨PCM格式详细说明 WAV文件是一种用于存储未压缩音频数据的文件格式,广泛应用于音频…...

【ES6复习笔记】数值扩展(16)

介绍 在 JavaScript 中&#xff0c;数值扩展提供了一些额外的功能&#xff0c;使得处理数值变得更加方便。本教程将介绍一些常用的数值扩展方法和属性。 1. Number.EPSILON Number.EPSILON 是 JavaScript 表示的最小精度。它的值接近于 2.2204460492503130808472633361816E-…...

百度热力图数据日期如何选择

目录 1、看日历2、看天气 根据研究内容定&#xff0c;一般如果研究城市活力的话&#xff0c;通常会写“非重大节假日&#xff0c;非重大活动&#xff0c;非极端天气等”。南方晴天不多&#xff0c;有小雨或者中雨都可认为没有影响&#xff0c;要不然在南方很难找到完全一周没有…...

Vue.js 高级组件开发:设计模式与实践

Vue.js 高级组件开发&#xff1a;设计模式与实践 引言一、组合式 API 与动态依赖注入1. 基于 provide/inject 的动态依赖2. 动态依赖注入与懒加载 二、动态渲染与自定义渲染函数1. 使用 Render 函数动态生成内容2. 自定义 vnode 操作 三、复杂场景下的动态表单生成与验证四、高…...

《一文读懂卷积网络CNN:原理、模型与应用全解析》

《一文读懂卷积网络CNN&#xff1a;原理、模型与应用全解析》 一、CNN 基本原理大揭秘&#xff08;一&#xff09;从人类视觉到 CNN 灵感&#xff08;二&#xff09;核心组件详解 二、经典 CNN 模型巡礼&#xff08;一&#xff09;LeNet-5&#xff1a;开山鼻祖&#xff08;二&a…...

MONI后台管理系统-数据敏感字段存储加密

前言&#xff1a;     在我们数据库中&#xff0c;存在很多的敏感数据&#xff0c;如用户表中&#xff0c;存在用户电话、身份证号、邮箱等属于用户的敏感信息&#xff0c;我们通常在存入数据库后&#xff0c;将其进行加密存储&#xff0c;以此来保证数据安全性。     …...

熟悉各类游戏设计模式的用途与限制,如 factory、strategy、mvc、object pool 等

良好的系统分析与设计能力要求开发者熟悉并正确运用各种设计模式来解决特定问题。设计模式是一种针对特定问题的通用解决方案&#xff0c;可提高代码的可复用性、可维护性和可扩展性。以下是对一些常见游戏设计模式的详细分析&#xff0c;包括其用途、限制和代码示例。 一、工厂…...

【RabbitMQ高级篇】消息可靠性问题(1)

目录 1.消息可靠性 1.1.生产者消息确认 1.1.1.修改配置 1.1.2.定义Return回调 1.1.3.定义ConfirmCallback 1.2.消息持久化 1.2.1.交换机持久化 1.2.2.队列持久化 1.2.3.消息持久化 1.3.消费者消息确认 1.3.1.演示none模式 1.3.2.演示auto模式 1.4.消费失败重试机制…...

ASP.NET |日常开发中常见问题归纳讲解

ASP.NET &#xff5c;日常开发中常见问题归纳讲解 前言一、性能问题1.1 数据库访问性能1.2 视图状态&#xff08;在ASP.NET Web Forms 中&#xff09; 二、安全问题2.1 SQL 注入2.2 跨站脚本攻击&#xff08;XSS&#xff09; 三、状态管理问题3.1 会话状态&#xff08;Session …...

【【深入浅出TinyRisc-v】】

深入浅出TinyRisc-v 本代码参考于 https://gitee.com/liangkangnan/tinyriscv 自己理解之后又重新写了一遍 tinyriscv.v // 涓嬮潰鏄鏁翠釜top妯″潡鐨勪功鍐? module tinyriscv(input clk ,input rst_n …...

常见的限流算法

常见的限流算法 限流的定义固定窗口算法滑动窗口算法漏桶算法&#xff08;推荐&#xff09;令牌桶算法(推荐)限流粒度本地限流&#xff08;单机限流&#xff09;分布式限流&#xff08;多机限流&#xff09;分布式限流的实现 限流的定义 限流&#xff0c;也称流量控制。是指系统…...

【Leetcode 每日一题】3159. 查询数组中元素的出现位置

问题背景 给你一个整数数组 n u m s nums nums&#xff0c;一个整数数组 q u e r i e s queries queries 和一个整数 x x x。 对于每个查询 q u e r i e s [ i ] queries[i] queries[i]&#xff0c;你需要找到 n u m s nums nums 中第 q u e r i e s [ i ] queries[i] q…...

xadmin后台首页增加一个导入数据按钮

xadmin后台首页增加一个导入数据按钮 效果 流程 1、在添加小组件中添加一个html页面 2、写入html代码 3、在urls.py添加导入数据路由 4、在views.py中添加响应函数html代码 <!DOCTYPE html> <html lang...

行为树详解(5)——事件驱动

【分析】 如果行为树的节点很多&#xff0c;那么会存在要经过很多节点才会走到动作节点的情况。显然&#xff0c;性能上不如状态机。 每帧都需要重新遍历一系列节点才会走到动作节点&#xff0c;而实际上很多条件节点在数帧内不会有变化&#xff0c;这是造成性能问题的重要原…...

3.若依前端项目拉取、部署、访问

因为默认RuoYi-Vue是使用的Vue2,所以需要另外去下载vue3来部署。 拉取代码 git clone https://gitee.com/ys-gitee/RuoYi-Vue3.git 安装node才能执行npm相关的命令 执行命令npm install 如果npm install比较慢的话&#xff0c;需要添加上国内镜像 npm install --registrhttp…...

Debian操作系统相对于Ubuntu有什么优势吗?

更高的稳定性&#xff1a;Debian 以其出色的稳定性闻名&#xff0c;得益于严格的软件包测试和发布流程。其稳定版经过长时间测试与验证&#xff0c;确保了系统的高度稳定&#xff0c;更适合对稳定性要求极高的长期运行服务器环境。而 Ubuntu 虽有稳定版本&#xff0c;但更新周期…...

【漏洞复现】CVE-2015-3337 Arbitrary File Reading

漏洞信息 NVD - CVE-2015-3337 Directory traversal vulnerability in Elasticsearch before 1.4.5 and 1.5.x before 1.5.2, when a site plugin is enabled, allows remote attackers to read arbitrary files via unspecified vectors. 在安装了具有“site”功能的插件以…...

win10、win11-鼠标右键还原、暂停更新

系统优化 win 10jihuo win 11jihuo鼠标右键还原暂停更新 update 2024.12.28win 10 jihuo winx&#xff0c;打开powershell管理员&#xff0c;输入以下命令,选择1并等待 irm https://get.activated.win | iex参考&#xff1a;https://www.bilibili.com/video/BV1TN411M72J/?sp…...

FFmpeg来从HTTP拉取流并实时推流到RTMP服务器

当使用FFmpeg来从HTTP拉取流并实时推流到RTMP服务器时&#xff0c;你可以使用以下命令&#xff1a; ffmpeg -i http://输入流地址 -c:v copy -c:a copy -f flv rtmp://RTMP服务器地址/应用名称/流名称 这是一个基本的命令示例&#xff0c;其中&#xff1a; - -i http://输入流地…...

<6>-MySQL表的增删查改

目录 一&#xff0c;create&#xff08;创建表&#xff09; 二&#xff0c;retrieve&#xff08;查询表&#xff09; 1&#xff0c;select列 2&#xff0c;where条件 三&#xff0c;update&#xff08;更新表&#xff09; 四&#xff0c;delete&#xff08;删除表&#xf…...

抖音增长新引擎:品融电商,一站式全案代运营领跑者

抖音增长新引擎&#xff1a;品融电商&#xff0c;一站式全案代运营领跑者 在抖音这个日活超7亿的流量汪洋中&#xff0c;品牌如何破浪前行&#xff1f;自建团队成本高、效果难控&#xff1b;碎片化运营又难成合力——这正是许多企业面临的增长困局。品融电商以「抖音全案代运营…...

基础测试工具使用经验

背景 vtune&#xff0c;perf, nsight system等基础测试工具&#xff0c;都是用过的&#xff0c;但是没有记录&#xff0c;都逐渐忘了。所以写这篇博客总结记录一下&#xff0c;只要以后发现新的用法&#xff0c;就记得来编辑补充一下 perf 比较基础的用法&#xff1a; 先改这…...

CocosCreator 之 JavaScript/TypeScript和Java的相互交互

引擎版本&#xff1a; 3.8.1 语言&#xff1a; JavaScript/TypeScript、C、Java 环境&#xff1a;Window 参考&#xff1a;Java原生反射机制 您好&#xff0c;我是鹤九日&#xff01; 回顾 在上篇文章中&#xff1a;CocosCreator Android项目接入UnityAds 广告SDK。 我们简单讲…...

linux 下常用变更-8

1、删除普通用户 查询用户初始UID和GIDls -l /home/ ###家目录中查看UID cat /etc/group ###此文件查看GID删除用户1.编辑文件 /etc/passwd 找到对应的行&#xff0c;YW343:x:0:0::/home/YW343:/bin/bash 2.将标红的位置修改为用户对应初始UID和GID&#xff1a; YW3…...

Spring Boot+Neo4j知识图谱实战:3步搭建智能关系网络!

一、引言 在数据驱动的背景下&#xff0c;知识图谱凭借其高效的信息组织能力&#xff0c;正逐步成为各行业应用的关键技术。本文聚焦 Spring Boot与Neo4j图数据库的技术结合&#xff0c;探讨知识图谱开发的实现细节&#xff0c;帮助读者掌握该技术栈在实际项目中的落地方法。 …...

土地利用/土地覆盖遥感解译与基于CLUE模型未来变化情景预测;从基础到高级,涵盖ArcGIS数据处理、ENVI遥感解译与CLUE模型情景模拟等

&#x1f50d; 土地利用/土地覆盖数据是生态、环境和气象等诸多领域模型的关键输入参数。通过遥感影像解译技术&#xff0c;可以精准获取历史或当前任何一个区域的土地利用/土地覆盖情况。这些数据不仅能够用于评估区域生态环境的变化趋势&#xff0c;还能有效评价重大生态工程…...

UR 协作机器人「三剑客」:精密轻量担当(UR7e)、全能协作主力(UR12e)、重型任务专家(UR15)

UR协作机器人正以其卓越性能在现代制造业自动化中扮演重要角色。UR7e、UR12e和UR15通过创新技术和精准设计满足了不同行业的多样化需求。其中&#xff0c;UR15以其速度、精度及人工智能准备能力成为自动化领域的重要突破。UR7e和UR12e则在负载规格和市场定位上不断优化&#xf…...

Android 之 kotlin 语言学习笔记三(Kotlin-Java 互操作)

参考官方文档&#xff1a;https://developer.android.google.cn/kotlin/interop?hlzh-cn 一、Java&#xff08;供 Kotlin 使用&#xff09; 1、不得使用硬关键字 不要使用 Kotlin 的任何硬关键字作为方法的名称 或字段。允许使用 Kotlin 的软关键字、修饰符关键字和特殊标识…...

有限自动机到正规文法转换器v1.0

1 项目简介 这是一个功能强大的有限自动机&#xff08;Finite Automaton, FA&#xff09;到正规文法&#xff08;Regular Grammar&#xff09;转换器&#xff0c;它配备了一个直观且完整的图形用户界面&#xff0c;使用户能够轻松地进行操作和观察。该程序基于编译原理中的经典…...