React 组件间通信方式简介


React 组件间通信主要分为以下四种情况:

  • 父组件向子组件通信
  • 子组件向父组件通信
  • 跨级组件之间通信
  • 非嵌套组件间通信

下面对这四种情况分别进行介绍:

 

父组件向子组件通信

父组件通过 props 和子组件进行通信,子组件得到 props 后进行相应的操作 
父组件 App.js:

 
  1. import React, { Component } from 'react';
  2. import './App.css';
  3. import Child1 from './Child1';
  4. class App extends Component {
  5. render() {
  6. return (
  7. <div className="App">
  8. <Child1 content="父组件传递的内容" />
  9. </div>
  10. );
  11. }
  12. }
  13. export default App;

子组件 Child1.jsx

 
  1. import React, { Component } from 'react';
  2. class Child1 extends Component {
  3. render () {
  4. return (
  5. <div>
  6. 子组件一:{ this.props.content }
  7. </div>
  8. )
  9. }
  10. }
  11. export default Child1;
 

子组件向父组件通信

和父组件向子组件通信类似,父组件通过传递 props 给子组件,只不过 props 的内容是一个函数,子组件通过调用父组件传递过来的回调函数,将子组件内容传递给父组件。 
父组件 App.js

 
  1. import React, { Component } from 'react';
  2. import logo from './logo.svg';
  3. import './App.css';
  4. import Child1 from './Child1';
  5. import Child2 from './Child2';
  6. class App extends Component {
  7. constructor(){
  8. super();
  9. this.state = {
  10. msg:'',
  11. }
  12. }
  13. childMsg(msg){
  14. this.setState({
  15. msg: msg,
  16. })
  17. }
  18. render() {
  19. return (
  20. <div className="App">
  21. 子组件二传过来的内容:{this.state.msg}
  22. <Child2 childMsg={this.childMsg.bind(this)}/>
  23. </div>
  24. );
  25. }
  26. }
  27. export default App;

子组件 Child2.jsx:

 
  1. import React, { Component } from 'react';
  2. class Child2 extends Component {
  3. msgHandle(){
  4. this.props.childMsg(this.refs.input.value);
  5. }
  6. render () {
  7. return (
  8. <div>
  9. 子组件二:
  10. <input type="text" placeholder='请输入内容' ref='input' />
  11. <button onClick={this.msgHandle.bind(this)}>点击向父组件传参</button>
  12. </div>
  13. )
  14. }
  15. }
  16. export default Child2;
 

跨级组件通信

跨级组件是指父组件向子组件的子组件进行通信,或者向更深层次的子组件通信,主要有两种方式:

  • 通过 props 层层传递
  • 使用 context 对象

对于层级不深的组件(三层以内),可以使用 props 进行层层传递,如果说层级更深的话, 
每一层组件都要去传递 props,并且这些 props 可能不是自身需要的,这就增加了复杂度,这种场景就可以使用 context 进行通信。context 是一个全局变量,相当于一个大容器,我们把要传递的信息放到这个容器里面,不管嵌套层级多深,子组件都可以获取到信息。使用 context 需要满足以下三个条件: 
1、父组件需要声明自己支持 context,并提供 context 对象中属性的 PropTypes 
2、子组件需要声明自己需要使用 context,并提供其需要使用的 context 属性的 PropTypes。 
3、父组件需要提供一个 getChildContext 函数,用来返回一个初始的 context 对象

props 层层传递: 
父组件App.js:

 
  1. import React, { Component } from 'react';
  2. import logo from './logo.svg';
  3. import './App.css';
  4. import Child1 from './Child1';
  5. class App extends Component {
  6. constructor(){
  7. super();
  8. }
  9. render() {
  10. return (
  11. <div className="App">
  12. <Child1 content="父组件传递给孙子组件的内容" />
  13. </div>
  14. );
  15. }
  16. }
  17. export default App;

Child1.jsx:

 
  1. import React, { Component } from 'react';
  2. import Child1_1 from './Child1_1';
  3. class Child1 extends Component {
  4. render () {
  5. return (
  6. <div>
  7. <Child1_1 content={this.props.content}/>
  8. </div>
  9. )
  10. }
  11. }
  12. export default Child1;

Child1_1.jsx:

 
  1. import React, { Component } from 'react';
  2. class Child1_1 extends Component {
  3. render () {
  4. return (
  5. <div>
  6. 子组件一的子组件:{ this.props.content }
  7. </div>
  8. )
  9. }
  10. }
  11. export default Child1_1;

context 对象传递 
父组件App.js:

 
  1. import React, { Component } from 'react';
  2. import logo from './logo.svg';
  3. import './App.css';
  4. import PropTypes from "prop-types";
  5. import Child1 from './Child1';
  6. class App extends Component {
  7. // 声明支持 context
  8. static childContextTypes = {
  9. msgs: PropTypes.string,
  10. callBack: PropTypes.func,
  11. }
  12. // 父组件提供一个函数,返回初始的 context 对象
  13. getChildContext(){
  14. return {
  15. msgs:'父组件传递的初始内容',
  16. callBack:this.callBack
  17. }
  18. }
  19. callBack(msgs){
  20. console.log(msgs);
  21. }
  22. render() {
  23. return (
  24. <div className="App">
  25. <Child1 />
  26. </div>
  27. );
  28. }
  29. }
  30. export default App;

子组件 Child1.jsx:

 
  1. import React, { Component } from 'react';
  2. import Child12 from './Child1_2';
  3. class Child1 extends Component {
  4. render () {
  5. return (
  6. <div>
  7. <Child12 />
  8. </div>
  9. )
  10. }
  11. }
  12. export default Child1;

子组件的子组件 Child1_2:

 
  1. import React, { Component } from 'react';
  2. import PropTypes from 'prop-types';
  3. class Child1_2 extends Component {
  4. // 子组件声明需要调动 context
  5. static contextTypes = {
  6. msgs:PropTypes.string,
  7. callBack:PropTypes.func
  8. }
  9. callBack(){
  10. this.context.callBack('孙子组件的信息');
  11. }
  12. render () {
  13. return (
  14. <div>
  15. 子组件一的子组件:{ this.context.msgs }
  16. <button onClick={this.callBack.bind(this)}>点击给爷爷组件传递信息</button>
  17. </div>
  18. )
  19. }
  20. }
  21. export default Child1_2;
 

非嵌套组件间通信

表示没有任何包含关系的组件,主要包括兄弟组件和不在同一个父级中的非兄弟组件。

  • 使用共同父组件的 context 对象通信
  • 使用自定义事件的方式

对于使用共同父组件 context 的方式会增加子组件和父组件之间的耦合度,对于层级较深的组件找到共同父组件也比较麻烦,但是这种方式可实施。

为了避免父子组件之间的耦合度,我们采用自定义事件的方式: 
需要安装 events 包,使用该模块的自定义事件机制npm i events --save 
在根目录下新建 events.js 文件:

 
  1. import { EventEmitter } from 'events';
  2. export default new EventEmitter();

父组件 App.js:

 
  1. import React, { Component } from 'react';
  2. import logo from './logo.svg';
  3. import './App.css';
  4. import Child1 from './Child1';
  5. import Child2 from './Child2';
  6. class App extends Component {
  7. render() {
  8. return (
  9. <div className="App">
  10. <Child1 />
  11. <Child2 />
  12. </div>
  13. );
  14. }
  15. }
  16. export default App;

子组件 Child1.jsx:

 
  1. import React, { Component } from 'react';
  2. import emitter from './events';
  3. class Child1 extends Component {
  4. constructor(props){
  5. super(props);
  6. this.state = {
  7. msg:''
  8. }
  9. }
  10. // 组件挂载完后,声明一个自定义事件
  11. componentDidMount(){
  12. this.eventEmitter = emitter.addListener('selfEventName',msg => {
  13. this.setState({
  14. msg: msg
  15. })
  16. })
  17. }
  18. // 组件销毁前清除事件监听
  19. componentWillUnmount(){
  20. emitter.removeListener(this.eventEmitter)
  21. }
  22. render () {
  23. return (
  24. <div>
  25. 子组件二传递过来的内容:{ this.state.msg }
  26. </div>
  27. )
  28. }
  29. }
  30. export default Child1;

子组件 Child2.jsx:

 
    1. import React, { Component } from 'react';
    2. import emitter from './events';
    3. class Child2 extends Component {
    4. msgHandle(){
    5. emitter.emit('selfEventName','自定义事件传参');
    6. }
    7. render () {
    8. return (
    9. <div>
    10. 子组件二:
    11. <button onClick={this.msgHandle.bind(this)}>点击向父组件传参</button>
    12. </div>
    13. )
    14. }
    15. }
    16. export default Child2;

React 组件间通信介绍的更多相关文章

  1. vue 和 react 组件间通信方法对比

    vue 和 react 组件间通信方法对比: 通信路径 vue的方法 react的方法 父组件 => 子组件 props(推荐).slot(推荐).this.$refs.this.$childr ...

  2. React组件间通信-sub/pub机制

    React生命周期第二个demo演示了兄弟组件的通信,需要通过父组件,比较麻烦:下面介绍sub/pub机制来事项组件间通信. 1.导包 npm i pubsub-js 2.UserSearch.jsx ...

  3. React 组件间通信 总结

    组件间通信 5.1.1. 方式一: 通过props传递 1)         共同的数据放在父组件上, 特有的数据放在自己组件内部(state) 2)         通过props可以传递一般数据和 ...

  4. React组件间通信

    众所周知,ReactJS组件与组件之间的通信是一个难点.在React实际开发中,父子组件之间的传值是比较常见的,刚入门的小伙伴很容易被组件之间的通信绕懵. 今天花了点时间总结了一下React父子组件之 ...

  5. react 组件间通信,父子间通信

    一.父组件传值给子组件 父组件向下传值是使用了props属性,在父组件定义的子组件上定义传给子组件的名字和值,然后在子组件通过this.props.xxx调用就可以了. 二.子组件传值给父组件 子组件 ...

  6. React 组件间通信

    https://jsfiddle.net/69z2wepo/9719/ <script src="https://facebook.github.io/react/js/jsfiddl ...

  7. React独立组件间通信联动

    React是现在主流的高效的前端框架,其官方文档 http://reactjs.cn/react/docs/getting-started.html 在介绍组件间通信时只给出了父子组件间通信的方法,而 ...

  8. React 精要面试题讲解(二) 组件间通信详解

    单向数据流与组件间通信 上文我们已经讲述过,react 单向数据流的原理和简单模拟实现.结合上文中的代码,我们来进行这节面试题的讲解: react中的组件间通信. 那么,首先我们把看上文中的原生js代 ...

  9. [转] React 中组件间通信的几种方式

    在使用 React 的过程中,不可避免的需要组件间进行消息传递(通信),组件间通信大体有下面几种情况: 父组件向子组件通信 子组件向父组件通信 跨级组件之间通信 非嵌套组件间通信 下面依次说下这几种通 ...

随机推荐

  1. java后台常用json解析工具问题小结

    若排版紊乱可查看我的个人博客原文地址 java后台常用json解析工具问题小结 这里不细究造成这些问题的底层原因,只是单纯的描述我碰到的问题及对应的解决方法 jackson将java对象转json字符 ...

  2. 怎么获取红米6 Pro的root权限

    红米6 Pro能有啥方法获得ROOT超级权限?做开发的人知道,android设备有ROOT超级权限,如果手机获得root相关权限,就能够实现更强大的功能,举例子,做开发的人部门的营销部门的妹子,使用一 ...

  3. Ganlia采样、统计及RRD记录周期(频次、间隔)的配置和更改

    Ganglia & RRD Ganglia是伯克利开发的一个集群监控软件.可以监视和显示集群中的节点的各种状态信息,比如如:cpu .mem.硬盘利用率, I/O负载.网络流量情况等,同时可以 ...

  4. C++ MySQL编程

    MySQL编程需要包含<mysql.h>头文件.该文件一般在MySQL安装目录下的include文件夹下. 包含头文件还不够,还需要包含“libmysql.lib”库,一般在lib文件夹下 ...

  5. 如何在linux环境上挂载磁盘

    1.1      fdisk -l 命令 查看可用的磁盘信息(如果没有显示可用的磁盘,可重启一下主机:reboot) 1.2      df –h 命令 可查看已挂的磁盘情况 1.3      pvs ...

  6. webpack学习笔记 ——篇2

    插件整理 extract-text-webpack-plugin 用于将css/less/sass等文件单独打包 https://webpack.docschina.org/plugins/extra ...

  7. [Paper][Link note]

    http://ieeexplore.ieee.org/document/6974670/

  8. RobotFramework之Run Keyword的使用

    RobotFramework之Run Keyword的使用        在之前写的RobotFramework(二)中有提到过这个Run Keyword关键字的使用,但是再做检查判断的时候,发现它的 ...

  9. js循环出相同name,不同id的按钮,对其进行点击回复操作

    function getseat(){ var option= "<button class='btn'style='margin:5px;' onclick='onclickSeat ...

  10. re正则表达式匹配字符串中的数字

    re.match(r'.*-(\d*).html',url_1).group(1) \d+匹配1次或者多次数字,注意这里不要写成*,因为即便是小数,小数点之前也得有一个数字:\.?这个是匹配小数点的, ...