摘要:函數屬性或者說事件在組件之間通信過程中是必不可少的,但是切莫讓它影響了大家對單向數據流這一概念的理解。這應該屬于一種的使用方式,而且這樣做有悖單向數據流原則。
上一篇文章 玩轉 React(六)- 處理事件 介紹了在 React 中如何處理用戶事件,以及 React 事件機制與原生 DOM 事件的差異和注意的問題,同時也介紹了事件處理函數中 this 的指向問題以及處理的幾種方式及其優(yōu)缺點。
大家在閱讀的過程中有任何為題可以給我留言,同時歡迎大家加入玩轉 React 微信群,我的微信號是 leobaba88,先加我好友,驗證信息:玩轉 React,然后我會拉你進群。
今天這篇文章要講的內容是關于多個組件之間如何共享數據,或者說是如何通信的。只有掌握了正確的組件之間通信的方式,才能在開發(fā)交互復雜的前端應用時做到游刃有余,所謂正確的方式也就是符合 React 設計理念的方式。使用一個框架時,一定要遵從框架的最佳實踐,人家框架是這樣設計的,你偏要那樣來用,用得不爽還要噴其不好用,那就不應該了。
內容摘要React 中的數據是單向自頂向下傳遞的。
單向數據流與雙向綁定的差異。
最符合 React 理念的組件之間共享數據的方式。
數據唯一來源原則。
一些不好的方式。
先跟 Redux 打個招呼。
其他一些關于組件間通信的內容(context、ref)。
組件之間通信的最佳方式現在我們就來探討下,什么樣的方式才是 React 中組件之間通信的正確方式。
在前面的文章中,我們有說過,React 之所以能勝任大型復雜前端項目的開發(fā),是因為其 單向數據流 這一重要特性,單向數據流能讓視圖更新邏輯變得簡單,從原始的對 DOM 操作變?yōu)閷祿僮?,簡單了就容易維護。
React 組件中數據的流動方向是自頂向下的,也就是說在組件樹中,數據只能從父組件以屬性的方式傳遞到子組件,父組件的數據可能是其接收到的屬性,也可能是自身的內部狀態(tài)。
有些同學這里可能會比較困惑,說子組件明明可以通過一個函數屬性將數據傳遞給父組件呀。好多同學甚至因此搞不明白單向數據流和雙向綁定的差異。其實換個角度考慮一下就清楚很多了,既然“數據傳遞”這個詞區(qū)分度不夠大,那就換個區(qū)分度比較大的說法。我們可以這樣理解,函數屬性是子組件用來通知父組件發(fā)生了什么,它更像是子組件觸發(fā)的一個事件,父組件可以依據業(yè)務邏輯來選擇如何處理這個事件,它可以更新數據后重新傳遞給子組件,也可以置之不理。
函數屬性(或者說事件)在組件之間通信過程中是必不可少的,但是切莫讓它影響了大家對單向數據流這一概念的理解。
數據雙向綁定不一樣,在雙向綁定中父組件將數據傳遞給子組件,子組件修改數據后會將數據回傳同步給父組件,父組件是無條件接受的。這里就不過多去說哪個好哪個差了,有興趣的同學可以自己去體會,懶一點的就堅持學習 React 吧。
狀態(tài)提升(Lifting State Up)既然 React 中的數據是單向自頂向下傳遞的,那么符合 React 這一特性的組件通信方式就顯而易見了。
狀態(tài)提升的意思是,當組件 A 需要依賴另外一個組件 B 的內部狀態(tài),而他們又不是父子關系時,需要將組件 B 的內部狀態(tài)提升到他們公共的祖先組件中管理。這樣他們就都可以通過屬性接收到這份數據了。
當組件 B 需要對數據進行變更時,可以通過函數屬性來通知祖先組件對數據更新,然后重新傳遞給子組件。
唯一數據來源(Single source of truth)有些同學可能又會迷惑,為什么多個組件之間必須要共用同一份數據,我可不可以引入一個事件庫,一個組件分發(fā)事件,另一個組件注冊相應的事件來接受數據自己維護。
類似的方案五花八門,會有很多,我認為這樣做當然是不好的,會有如下問題:
破壞了組件的封裝性,易于復用的組件都是相對獨立的,它只需要定義自己需要的數據和行為(函數屬性)即可,我不需要誰幫我分發(fā)事件。
數據傳遞是不連續(xù)的,這樣做會增加項目的復雜性,當項目到一定階段后,對這份數據的依賴就變得千絲萬縷、難以維護了。
相同的數據會有多個副本,需要保證數據同步,在增加項目復雜性的同時也提高了出現BUG的幾率。
這是我個人的看法,我也確實有遇到過這種用法,有不同意見大家可以進群交流。
數據唯一來源是官方推薦的數據共享的原則,也是最符合 React 設計理念,與單向數據流特性相輔相成的,希望大家務必遵守。
ReduxRedux 是一個狀態(tài)管理庫,它不是專屬于 React 技術棧的,但是跟 React 配合起來相當不錯。
當我們的前端應用規(guī)模較小的時候,我們可以不引入任何的狀態(tài)管理工具,只需要依據上面說的狀態(tài)提升的方式來管理應用狀態(tài)即可。為了讓應用的狀態(tài)更直觀,你可以將跟組件作為狀態(tài)總線,來管理整個應用所有的狀態(tài)。而且對于小規(guī)模的項目是推薦這樣來做的,沒有必要高射炮打蚊子,過渡設計。
但是當前端應用規(guī)模變得比較復雜時,我們就需要有類似 Redux 這樣一個來專門進行狀態(tài)管理的東西了。它的職責如下:
維護一個數據倉庫(store)管理整個應用的狀態(tài)(state),確保數據的唯一來源。
可以通過 dispatch 方法分發(fā)一個 action,來通知 Redux 需要對數據進行變更。
Redux 接收到 action 后可以依據 action 的類型對 state 進行相應的修改。
數據跟新后 Redux 會觸發(fā)注冊的監(jiān)聽器(如:更新組件屬性),完成視圖更新。
Redux 跟 React 一起來用,更詳細的介紹可以參考:官方文檔,這里大家可以先簡單了解下,在后面關于 React 實戰(zhàn)的文章中也會詳細介紹 Redux 的使用。
類似的狀態(tài)管理工具還有:MobxJS,感興趣的同學也可以了解下。
關于組件通信的其他內容在 React 中還有一些其他的與組件間通信相關的知識,這里也順便跟大家介紹下。
context首先說一下,這是一個不推薦使用的特性,React 官方有明確說明,這是一個實驗性的API,可能會在后面的版本中去掉這個東西。所以我是從來不用的,呵呵!
context 的作用是啥呢,當大家有過 React 實戰(zhàn)經驗時,很容易遇到這種場景,如果組件的層級組織得不合適,可能會嵌套的非常深,當底層的一個組件需要使用頂層一個組件的數據時,需要通過屬性一層層傳遞下去,非常繁瑣。
context 就是解決這個問題的,只需要在頂層組件中聲明 context,那它的所有子組件可以通過 this.context 直接獲取得到。如下實例所示:
import React from "react"; import PropTypes from "prop-types"; class Button extends React.Component { render() { return ( ); } } Button.contextTypes = { color: PropTypes.string }; class Message extends React.Component { render() { return ({this.props.text}); } } class MessageList extends React.Component { getChildContext() { return {color: "purple"}; } render() { const children = this.props.messages.map((message) =>); return {children}; } } MessageList.childContextTypes = { color: PropTypes.string };
實例中,組件層級關系是:MessageList -> Message -> Button。
MessageList 組件中維護一個 color 值用于 Button 組件的背景色,一般情況下我們需要將 color 以屬性的方式傳給 Message 組件,再通過屬性傳給 Button 組件。然后在實例中,通過 React 的 context 功能,MessageList 可以將 color 的值越過 Message 直接傳給 Button。
是不是很方便?確實很方便,但是這會導致數據傳遞不連續(xù),過度使用會使得項目邏輯變得不直觀,增加項目維護的復雜性。
ref每一個 React 組件有一個特殊的屬性 ref,該屬性的值可以是一個字符串,也可以是一個函數。由于字符串形式的 ref 在內部實現和實際使用中存在諸多問題,官方不推薦使用,而且可能在未來的版本中會移除,所以我們也沒必要聊它了,只要大家在看到字符串形式的 ref 屬性時知道也有這種用法就可以了。
當 ref 屬性值是一個函數時,如果組件是一個 HTML 元素兼容的 React 內部組件時(如:div、img 等),函數接收其對應的原生 DOM 節(jié)點作為參數。如果組件是一個我們以類的方式定義的組件時,函數接收該組件類的實例作為參數。需要注意的是,如果組件是一個以函數的方式定義的組件,那么設置為 ref 值得函數永遠都會接收到一個 null。
那么 ref 與組件之間的通信有什么關系呢?請看上段文字加粗內容和下面這個實例:
class UserForm extends React.Component { constructor(props) { super(props) this.state = { name: null, age: null } } formData() { return this.state } handleFieldChange(e) { const { name, value } = e.target this.setState({ [name]: value }) } render() { return (this.handleFieldChange(e)} /> this.handleFieldChange(e)} />) } } class App extends React.Component { handleSubmit() { const formData = this.form.formData() alert(`formData: ${JSON.stringify(formData)}`) } render() { return () } } ReactDOM.render({this.form = form}} /> , document.querySelector("#root"))
演示地址:https://codepen.io/Sarike/pen...
既然通過 ref 能夠獲取子組件的實例,那么我們自然可以調用其成員方法,從而獲取數據。
當然,目前這確實能工作,但絕對不是一種好的方式。因為作為一個組件,是需要有一定的封裝性的,它應該對外只會承諾我接受什么樣的屬性,而不會承諾有什么樣的成員方法。換句話說,如果 JavaScript 的類支持私有成員方法,那么 React 組件類中的成員方法都應該定義成私有的。
這應該屬于一種 Hack 的使用方式,而且這樣做有悖單向數據流原則。
ref 有它自己的使用場景,這里只是說明這種方式不適用于組件之間通信。
總結雖然啰嗦了這么多,實際上只希望大家知道一件事情,請使用狀態(tài)提升的方式在多個組件之間共享數據,切記維持應用單向數據流和數據唯一來源原則。
文章中有些觀點仁者見仁,有什么疑惑歡迎留言討論。
好久沒更新了,但是沒有放棄,感謝大家支持。歡迎加我微信好友:leobaba88,進群交流。驗證信息:玩轉 React。
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉載請注明本文地址:http://systransis.cn/yun/90330.html
摘要:本人計劃編寫一個針對中初級前端開發(fā)者學習的系列教程玩轉。使用的原因是新的語言規(guī)范開發(fā)效率更高代碼更優(yōu)雅,尤其是基于開發(fā)的項目。其次也是目前特別流行的一個前端框架,截止目前,上有將近萬,國內一二線互聯(lián)網公司都有深度依賴開發(fā)的項目。 本人計劃編寫一個針對中初級前端開發(fā)者學習 React 的系列教程 - 《玩轉 React》。 文章更新頻率:每周 1 ~ 2 篇。 目錄 玩轉 React(...
摘要:類組件中的增加學習成本,類組件在基于現有工具的優(yōu)化上存在些許問題。由于業(yè)務變動,函數組件不得不改為類組件等等。那么可愛的各位看官,還不趕緊使用起來在線示例點我版本基礎入門項目錄像教程 視圖與業(yè)務,好一對冤家 業(yè)務型model model是需要精心的設計和合理的劃分的,這是我們之前開發(fā)大型的redux+react單頁面應用,大家都認同的真理,同樣的,在react-control-cent...
摘要:這也就是所謂的單向數據流,在這種開發(fā)方式下,會讓你更新視圖的邏輯非常清晰簡單,哪怕你的前端交互很復雜,也不至于讓你的代碼那么容易變成一坨。就是在前端開發(fā)過程中,要善于觀察和抽象。 這是《玩轉 React》系列的第二篇。在該篇中,我們來了解下,React 的出現到底給我們的開發(fā)方式帶來了什么樣的變化。 我的感觸可以用一個字來形容,爽!主要爽在以下兩個方面。 視圖是數據的映射(單向數據流)...
摘要:屬性是一個組件的外部輸入。只會在開發(fā)模式下進行屬性類型檢查,當代碼進行生產發(fā)布后,為了減少額外的性能開銷,類型檢查將會被略過。某個類的實例枚舉,屬性值必須為其中的某一個值。屬性為一個數組,且數組中的元素必須符合指定類型。 在第二篇文章 《新型前端開發(fā)方式》 中有說到 React 有很爽的一點就是給我們一種創(chuàng)造 HTML 標簽的能力,那么今天這篇文章就詳細講解下 React 是如何提供這...
摘要:屬性是一個組件的外部輸入。只會在開發(fā)模式下進行屬性類型檢查,當代碼進行生產發(fā)布后,為了減少額外的性能開銷,類型檢查將會被略過。某個類的實例枚舉,屬性值必須為其中的某一個值。屬性為一個數組,且數組中的元素必須符合指定類型。 在第二篇文章 《新型前端開發(fā)方式》 中有說到 React 有很爽的一點就是給我們一種創(chuàng)造 HTML 標簽的能力,那么今天這篇文章就詳細講解下 React 是如何提供這...
閱讀 2126·2021-11-23 09:51
閱讀 3712·2021-10-20 13:49
閱讀 1718·2021-09-06 15:13
閱讀 1828·2021-09-06 15:02
閱讀 3181·2021-09-02 15:11
閱讀 899·2019-08-29 15:37
閱讀 1744·2019-08-29 13:24
閱讀 2283·2019-08-29 11:28