React 組件間的通信

前言

? 從官網(wǎng)上也有介紹組件間如何通信,但不夠詳細(xì),這里做個(gè)小結(jié),方便對(duì)比和回顧
?

本文內(nèi)容

? 處理組件之間的通信, 主要取決于組件之間的關(guān)系,因此我們劃分為以下三種:

  1. 【父組件】向【子組件】傳值;
  2. 【子組件】向【父組件】傳值;
  3. 【組件A】向無關(guān)系【組件B】傳值,一般為兄弟組件;
    ?

一、「父組件」向「子組件」傳值

? 這是最普遍的用法,實(shí)現(xiàn)上也非常簡(jiǎn)單,主要是利用props來實(shí)現(xiàn)

// 父組件
import React from 'react';
import Son from './components/son';
class Father extends React.Component {
    constructor(props) {
        // 這里要加super,否則會(huì)報(bào)錯(cuò)
        super(props);
        this.state = {
            checked: true
        }
    }

    render() {
        return (
            <Son text="Toggle me" checked={this.state.checked} />
        )
    }
}
// 子組件
class Son extends React.Component {
    render() {
        // 接收來自父組件的參數(shù)
        let checked = this.props.checked,
            text = this.props.text;
        return (
            <label>{text}: <input type="checkbox" checked={checked} /></label>
        )
    }
}

? 多想一點(diǎn):

?如果組件的嵌套層次太多,那么從外到內(nèi)的交流成本就會(huì)加深,通過 props 傳值的優(yōu)勢(shì)就不明顯,因此,我們還是要盡可能的編寫結(jié)構(gòu)清晰簡(jiǎn)單的組件關(guān)系, 既也要遵循組件獨(dú)立原則,又要適當(dāng)控制頁面,不可能或極少可能會(huì)被單用的代碼片,可不編寫成一個(gè)子組件
?

二、「子組件」向「父組件」傳值

? 我們知道,react的數(shù)據(jù)控制分為兩種,為 propsstate;其中,props 如上剛介紹過,它是父組件向子組件傳值時(shí)作為保存參數(shù)的數(shù)據(jù)對(duì)象;而 state 是組件存放自身數(shù)據(jù)的數(shù)據(jù)對(duì)象。這兩者最主要的區(qū)別就是,props屬于父組件傳給子組件的只讀數(shù)據(jù),在子組件中不能被修改,而state在自身組件中使用時(shí),可以通過setState來修改更新。

? 子組件向父組件傳值,需要控制自己的state,并發(fā)起父組件的事件回調(diào)來通知父組件

// 父組件
import Son from './components/son';
class Father extends React.Component {
    constructor(props) {
        super(props)
        this.state = {
            checked: false
        }
    }
    onChildChanged() {
        this.setState({
            checked: newState
        })
    }

    render() {
        let isChecked = this.state.checked ? 'yes' : 'no';
        return (
            <div>
                <span>Are you checked: {isChecked }</span>
                <Son text="Toggle me" 
                      initialChecked={this.state.checked}
                      callbackParent={this.onChildChanged.bind(this)}
                 ></Son>
            </div>
        )
    }
}
// 子組件
class Son extends React.Component {
    constructor(props) {
        super(props);
        this.state = {
            checked: this.props.initialChecked
        }
    }
    onTextChange() {
        let newState = !this.state.check.checked;
        this.setState({
            checked: newState
        });
        // 注意,setState 是一個(gè)異步方法,state值不會(huì)立即改變,回調(diào)時(shí)要傳緩存的當(dāng)前值,     
        // 也可以利用傳遞一個(gè)函數(shù)(以上傳的是對(duì)象),并傳遞prevState參數(shù)來實(shí)現(xiàn)數(shù)據(jù)的同步更新
        this.props.callbackParent(newState);
    }
    render() {
        let text= this.props.text;
        let checked = this.state.checked;
        return (
            <label>{text}: <input type="checkbox" checked={checked}  onChange={this.onTextChange.bind(this)}></label>
        )
    }
}

多想一點(diǎn):

  1. 同樣應(yīng)當(dāng)避免深層次的組件嵌套
  2. 這里其實(shí)是依賴了props來傳遞事件的引用,并通過回調(diào)的方式來實(shí)現(xiàn),在沒有使用工具情況下,可以使用該辦法

拓展一點(diǎn):
? 在onChange 事件或者其他React事件中,你能獲取以下信息:

  1. this」 指向你的組件
  2. 一個(gè)參數(shù)」 一個(gè)react合成事件, SyntheticEvent
    我們知道,React對(duì)所有事件的管理都是自己封裝實(shí)現(xiàn)的,html中的 onclick 被封裝成了 onClick, onchange 被封裝成了 onChange。從根本上來說,他們都是被綁定在body上的。
    ?

多個(gè)子組件回調(diào)同一個(gè)回調(diào)函數(shù)情況

? 父組件中大概率包含多個(gè)子組件,為節(jié)省和簡(jiǎn)潔代碼,遵循 don't repeat yourself 原則,我們會(huì)讓一個(gè)回調(diào)函數(shù)實(shí)現(xiàn)多個(gè)子組件的功能,或多個(gè)組件協(xié)作完成指定功能

import React from 'react';
import Son from './components/son';
class Father extends React.Componnet {
    constructor(props) {
        super(props);
        this.state =  {
            totalChecked: 0
        }
    }
    onChildChangeed() {
        let newTotal = this.state.totalChecked + (new State ? 1 : -1 );
        this.setState({
             totalChecked = this.state.totalChecked;
        });
    }
    render() {
        return (
            <div>
                <div>Checked numbers: {this.state.totalChecked}</div>
                <Son text="Toggle me" initialChecked={this.state.checked} callbackParent={this.onChildChanged} />
                <Son text="Toggle me too" initialChecked={this.state.checked} callbackParent={this.onChildChanged} />
                 <Son text="Add me" initialChecked={this.state.checked} callbackParent={this.onChildChanged} />
            </div>
        )
    }
}
// 子組件
class Son extends React.Component {
    constructor(props) {
        super(props);
        this.state = {
            checked: this.props.initialChecked
        }
    } 

    onTextChange() {
        let newState = !this.state.checked;
        this.setState({
            checked: newState
        })
        // setState異步方法問題,注意傳值
        this.props.callbackParent(newState);
    }

    render() {
        let text = this.props.checked;
        let checked = this.state.checked;
        return {
            <label>{text}: <input type="checkbox" checked={checked} onChange={this.onTextChange.bind(this)} /></label>
        }
    }
}

? 多想一點(diǎn):
? 在本案例中,我們引用了三個(gè) Son 子組件, 每個(gè) Son 組件都獨(dú)立工作互不干擾,該例中,增加了一個(gè) totalChecked 來替代之前的 checked, 當(dāng)組件觸發(fā)onTextChange 后,觸發(fā)父組件的回調(diào)函數(shù)使得父組件的值得以改變。
?

三、組件A和無關(guān)系組件B之間的通信

? 如果組件之間沒有任何關(guān)系,或者組件嵌套的層次比較深,或者,你為了一些組件能夠訂閱,寫入一些信號(hào),不想讓兩個(gè)組件之間插入一個(gè)組件,而是讓兩個(gè)組件出于獨(dú)立的關(guān)系。對(duì)于時(shí)間系統(tǒng),有兩個(gè)基本操作:

  1. 訂閱: subscribe
  2. 監(jiān)聽: listen
    并發(fā)送 send / 觸發(fā) trigger / 發(fā)布 publish / 發(fā)送 dispatch 通知那些想要的組件

1. Event Emitter/Target/Dispatcher

? 特點(diǎn): 需要一個(gè)指定的訂閱源

// to subscribe
otherObiect.addEventListener('clickEvent', function() {
    alert('click!');
})
// to dispatch
this.dispatchEvent('clickEvent');

2. Publish / Subscribe

? 特點(diǎn): 觸發(fā)的時(shí)候,不需要指定一個(gè)特定的源,使用全局對(duì)象廣播的方式來處理事件

// to subscribe
globalBroadcaster.subcribe('clickEvent', function() {
    alert('cilck!');    
})
// to publish
globalBroadcaster.publish('clickEvent');

? 這種方案還有一個(gè)插件可用, 即 PubSubJs;用法如下:

import Pubsub from 'pubsub-js';
...
// to subscribe
Pubsub.subscribe('EVENT', (msg, param) => {
    console.log(msg, param);
});
// to publish
Pubsub.publish('EVENT', param);

3. Single

? 特點(diǎn): 與 Event Emitter/Target/Dispatcher 類似,但是不要使用隨機(jī)字符串作為事件觸發(fā)的引用。觸發(fā)事件的每一個(gè)對(duì)象都需要一個(gè)確切的名字,并且在觸發(fā)的時(shí)候,也必須要指定確切的事件

// to subscribe
otherObject.clicked.add(function() {
    alert('click');
})
// to dispatch
this.clicked.dispatch();

? React 團(tuán)隊(duì)使用的是:js-signals 它基于 Signals 模式,用起來相當(dāng)不錯(cuò)。
?

事件訂閱與取消

? 使用React事件的時(shí)候,必須關(guān)注以下兩個(gè)方法:

  1. componentDidMount
  2. componentWillUnmount

? 在 componentDidMount 事件中,等待組件掛載 mounted 完成,再訂閱事件;訂閱的事件需要在組件卸載 componentWillUnmount 的時(shí)候取消事件的訂閱。

? 因?yàn)榻M件的渲染和銷毀是有 React 來控制的,我們不知道怎么引用他們,所以EventEmitter 模式在處理事件的時(shí)候用處不大,Pub/Sub 模式就好用些,因?yàn)槲覀儾恍枰酪迷谀摹?br> ?

ES6策略: yield and js-csp

? ES6中有一種傳遞信息的方式,使用生成函數(shù) generatorsyield 關(guān)鍵字,用法參考以下例子

import csp from 'js-csp';

function* list() {
    for(var i = 0; i< arguments.length; i++) {
        yield arguments[i];
    }
    return "done";
}
var o = list(1, 2, 3);
var cur = o.next;
while (!cur.done) {
    cur = o.next();
    console.log(cur);
}

?

結(jié)束語

? 數(shù)據(jù)在組件中應(yīng)該以什么樣的方式進(jìn)行傳遞取決于組件之間存在什么樣的關(guān)系和當(dāng)時(shí)的業(yè)務(wù)場(chǎng)景需求,大家應(yīng)該根據(jù)項(xiàng)目合理選擇數(shù)據(jù)處理的方案,這很可能減少你大量的代碼量和代碼邏輯。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • 在使用 React 的過程中,不可避免的需要組件間進(jìn)行消息傳遞(通信),組件間通信大體有下面幾種情況: 父組件向子...
    柏丘君閱讀 93,438評(píng)論 7 80
  • 說在前面 關(guān)于 react 的總結(jié)過去半年就一直碎碎念著要搞起來,各(wo)種(tai)原(lan)因(le)。心...
    陳嘻嘻啊閱讀 7,018評(píng)論 7 41
  • 目前,react組件有三種寫法,分別是es5的createClass寫法,es6的class寫法,以及statel...
    ZoomFunc閱讀 1,908評(píng)論 0 1
  • 課堂要善于給學(xué)生留白,要聯(lián)系生活的實(shí)際,鼓勵(lì)學(xué)生大膽的思考與質(zhì)疑,否則,課堂教學(xué)勢(shì)必成為“一個(gè)人的狂歡,一群人...
    王曉峰Dev閱讀 342評(píng)論 3 12
  • 我貪婪地嗅著文字散發(fā)出來的宛如八月桂花馥郁的芳香,深情款款地邁出雙腿走向他,他儼然我玉樹臨風(fēng),顧盼生姿的情郎。在我...
    移云藏月閱讀 1,066評(píng)論 17 30

友情鏈接更多精彩內(nèi)容