前言
在使用 React 的过程中,不可避免的需要组件间进行消息传递(通信),组件间通信大体有下面几种情况:
父组件通过向子组件传递 props,子组件得到 props 后进行相应的处理
1 | // Child.jsx |
1 | // Parent.jsx |
2. 子组件向父组件通信
利用回调函数
利用自定义事件机制
1 | // List3.jsx |
1 | // App.jsx |
3. 跨级组件通信
层层组件传递props
例如A组件和B组件之间要进行通信,先找到A和B公共的父组件,A先向C组件通信,C组件通过props和B组件通信,此时C组件起的就是中间件的作用
使用context
context是一个全局变量,像是一个大容器,在任何地方都可以访问到,我们可以把要通信的信息放在context上,然后在其他组件中可以随意取到;
但是React官方不建议使用大量context,尽管他可以减少逐层传递,但是当组件结构复杂的时候,我们并不知道context是从哪里传过来的;而且context是一个全局变量,全局变量正是导致应用走向混乱的罪魁祸首.
使用context
下面例子中的组件关系: ListItem是List的子组件,List是app的子组件
1 | // ListItem.jsx |
1 | // List.jsx |
1 | // App.jsx |
4. 没有嵌套关系的组件通信
使用自定义事件机制
在componentDidMount事件中,如果组件挂载完成,再订阅事件;在组件卸载的时候,在componentWillUnmount事件中取消事件的订阅;
以常用的发布/订阅模式举例,借用Node.js Events模块的浏览器版实现
使用自定义事件的方式
下面例子中的组件关系: List1和List2没有任何嵌套关系,App是他们的父组件;
实现这样一个功能: 点击List2中的一个按钮,改变List1中的信息显示
首先需要项目中安装events包:
1 | npm install events --save |
在src下新建一个util目录里面建一个events.js
1 | import { EventEmitter } from 'events'; |
1 | // list1.jsx |
1 | // List2.jsx |
1 | // APP.jsx |
自定义事件是典型的发布订阅模式,通过向事件对象上添加监听器和触发事件来实现组件之间的通信
总结
父组件向子组件通信: props
子组件向父组件通信: 回调函数/自定义事件
跨级组件通信: 层层组件传递props/context
没有嵌套关系组件之间的通信: 自定义事件
ps:
在进行组件通信的时候,主要看业务的具体需求,选择最合适的; 当业务逻辑复杂到一定程度,就可以考虑引入Mobx,Redux等状态管理工具