翻譯:劉小夕原文連接:https://dmitripavlutin.com/7-...javascript
原文的篇幅很是長,不過內容太過於吸引我,仍是忍不住要翻譯出來。此篇文章對編寫可重用和可維護的React組件很是有幫助。但由於篇幅實在太長,我對文章進行了分割,本篇文章重點闡述 封裝
。因本人水平有限,文中部分翻譯可能不夠準確,若是您有更好的想法,歡迎在評論區指出。java
更多文章可戳: https://github.com/YvetteLau/...react
———————————————我是一條分割線————————————————git
一個封裝組件提供
props
控制其行爲而不是暴露其內部結構。
耦合是決定組件之間依賴程度的系統特性。根據組件的依賴程度,可區分兩種耦合類型:github
鬆耦合是咱們設計應用結構和組件之間關係的目標。數組
鬆耦合應用(封裝組件)
鬆耦合會帶來如下好處:數據結構
相反,緊耦合的系統會失去上面描述的好處。主要缺點是很難修改高度依賴於其餘組件的組件。即便是一處修改,也可能致使一系列的依賴組件須要修改。app
緊耦合應用(組件無封裝)
封裝 或 信息隱藏 是如何設計組件的基本原則,也是鬆耦合的關鍵。異步
封裝良好的組件隱藏其內部結構,並提供一組屬性來控制其行爲。函數
隱藏內部結構是必要的。其餘組件不必知道或也不依賴組件的內部結構或實現細節。
React
組件多是函數組件或類組件、定義實例方法、設置 ref
、擁有 state
或使用生命週期方法。這些實現細節被封裝在組件內部,其餘組件不該該知道這些細節。
隱藏內部結構的組件彼此之間的依賴性較小,而下降依賴度會帶來鬆耦合的好處。
細節隱藏是隔離組件的關鍵。此時,你須要一種組件通訊的方法:props
。porps
是組件的輸入。
建議 prop
的類型爲基本數據(例如,string
、 number
、boolean
):
<Message text="Hello world!" modal={false} />;
必要時,使用複雜的數據結構,如對象或數組:
<MoviesList items={['Batman Begins', 'Blade Runner']} />
prop
能夠是一個事件處理函數和異步函數:
<input type="text" onChange={handleChange} />
prop
甚至能夠是一個組件構造函數。組件能夠處理其餘組件的實例化:
function If({ component: Component, condition }) { return condition ? <Component /> : null; } <If condition={false} component={LazyComponent} />
爲了不破壞封裝,請注意經過 props
傳遞的內容。給子組件設置 props
的父組件不該該暴露其內部結構的任何細節。例如,使用 props
傳輸整個組件實例或 refs
都是一個很差的作法。
訪問全局變量一樣也會對封裝產生負面影響。
組件的實例和狀態對象是封裝在組件內部的實現細節。所以,將狀態管理的父組件實例傳遞給子組件會破壞封裝。
咱們來研究一下這種狀況。
一個簡單的應用程序顯示一個數字和兩個按鈕。第一個按鈕增長數值,第二個按鈕減小數值:
這個應用由兩個組件組成:<App>
和 <Controls>
.
number
是 <App>
的 state
對象,<App>
負責 將這個數字渲染到頁面。
// 問題: 封裝被破壞 class App extends Component { constructor(props) { super(props); this.state = { number: 0 }; } render() { return ( <div className="app"> <span className="number">{this.state.number}</span> <Controls parent={this} /> </div> ); } }
<Controls>
負責渲染按鈕,併爲其設置事件處理函數,當用戶點擊按鈕時,父組件的狀態將會被更新:number
加1或者減1((
updateNumber()方法`)
// 問題: 使用父組件的內部結構 class Controls extends Component { render() { return ( <div className="controls"> <button onClick={() => this.updateNumber(+1)}> Increase </button> <button onClick={() => this.updateNumber(-1)}> Decrease </button> </div> ); } updateNumber(toAdd) { this.props.parent.setState(prevState => ({ number: prevState.number + toAdd })); } }
當前的實現有什麼問題?
<App>
的封裝被破壞,由於它的內部結構在應用中傳遞。<App>
錯誤地容許 <Controls>
直接去修改其 state
。Controls
知道了太多父組件 <App>
的內部細節,它能夠訪問父組件的實例,知道父組件是一個有狀態組件,知道父組件的 state
對象的細節(知道 number
是父組件 state
的屬性),而且知道怎麼去更新父組件的 state
.一個麻煩的結果是: <Controls>
將很難測試和重用。對 <App>
結構的細微修改會致使須要對 <Controls>
進行修改(對於更大的應用程序,也會致使相似耦合的組件須要修改)。
解決方案是設計一個方便的通訊接口,考慮到鬆耦合和封裝。讓咱們改進兩個組件的結構和屬性,以便恢復封裝。
只有組件自己應該知道它的狀態結構。<App>
的狀態管理應該從 <Controls>
(updateNumber()
方法)移到正確的位置:即 <App>
組件中。
<App>
被修改成 <Controls>
設置屬性 onIncrease
和 onDecrease
。這些是更新 <App>
狀態的回調函數:
// 解決: 恢復封裝 class App extends Component { constructor(props) { super(props); this.state = { number: 0 }; } render() { return ( <div className="app"> <span className="number">{this.state.number}</span> <Controls onIncrease={() => this.updateNumber(+1)} onDecrease={() => this.updateNumber(-1)} /> </div> ); } updateNumber(toAdd) { this.setState(prevState => ({ number: prevState.number + toAdd })); } }
如今,<Controls>
接收用於增長和減小數值的回調,注意解耦和封裝恢復時:<Controls>
再也不須要訪問父組件實例。也不會直接去修改父組件的狀態。
並且,<Controls>
被修改成了一個函數式組件:
// 解決方案: 使用回調函數去更新父組件的狀態 function Controls({ onIncrease, onDecrease }) { return ( <div className="controls"> <button onClick={onIncrease}>Increase</button> <button onClick={onDecrease}>Decrease</button> </div> ); }
<App>
組件的封裝已經恢復,狀態由其自己管理,也應該如此。
此外,<Controls>
不在依賴 <App>
的實現細節,onIncrease
和 onDecrease
在按鈕被點擊的時候調用,<Controls>
不知道(也不該該知道)這些回調的內部實現。
<Controls>
組件的可重用性和可測試性顯著增長。
<Controls>
的複用變得很容易,由於它除了須要回調,沒有其它依賴。測試也變得簡單,只需驗證單擊按鈕時,回調是否執行。
最後謝謝各位小夥伴願意花費寶貴的時間閱讀本文,若是本文給了您一點幫助或者是啓發,請不要吝嗇你的贊和Star,您的確定是我前進的最大動力。https://github.com/YvetteLau/...
推薦關注本人公衆號