摘要:精讀源碼一共行,我們分析一下其精妙的方式。更多討論討論地址是精讀新用法如果你想參與討論,請點擊這里,每周都有新的主題,周末或周一發布。前端精讀幫你篩選靠譜的內容。
1 引言
很高興這一期的話題是由 epitath 的作者 grsabreu 提供的。
前端發展了 20 多年,隨著發展中國家越來越多的互聯網從業者涌入,現在前端知識玲瑯滿足,概念、庫也越來越多。雖然內容越來越多,但作為個體的你的時間并沒有增多,如何持續學習新知識,學什么將會是個大問題。
前端精讀通過吸引優質的用戶,提供最前沿的話題或者設計理念,雖然每周一篇文章不足以概括這一周的所有焦點,但可以保證你閱讀的這十幾分鐘沒有在浪費時間,每一篇精讀都是經過精心篩選的,我們既討論大家關注的焦點,也能找到倉庫角落被遺忘的珍珠。
2 概述在介紹 Epitath 之前,先介紹一下 renderProps。
renderProps 是 jsx 的一種實踐方式,renderProps 組件并不渲染 dom,但提供了持久化數據與回調函數幫助減少對當前組件 state 的依賴。
RenderProps 的概念react-powerplug 就是一個 renderProps 工具庫,我們看看可以做些什么:
{({ on, toggle }) => }
Toggle 就是一個 renderProps 組件,它可以幫助控制受控組件。比如僅僅利用 Toggle,我們可以大大簡化 Modal 組件的使用方式:
class App extends React.Component { state = { visible: false }; showModal = () => { this.setState({ visible: true }); }; handleOk = e => { this.setState({ visible: false }); }; handleCancel = e => { this.setState({ visible: false }); }; render() { return (); } } ReactDOM.render(Some contents...
Some contents...
Some contents...
, mountNode);
這是 Modal 標準代碼,我們可以使用 Toggle 簡化為:
class App extends React.Component { render() { return ({({ on, toggle }) => ( ); } } ReactDOM.render()} Some contents...
Some contents...
Some contents...
, mountNode);
省掉了 state、一堆回調函數,而且代碼更簡潔,更語義化。
renderProps 內部管理的狀態不方便從外部獲取,因此只適合保存業務無關的數據,比如 Modal 顯隱。RenderProps 嵌套問題的解法
renderProps 雖然好用,但當我們想組合使用時,可能會遇到層層嵌套的問題:
{counter => { {toggle => { ; }}; }}
因此 react-powerplugin 提供了 compose 函數,幫助聚合 renderProps 組件:
import { compose } from "react-powerplug" const ToggleCounter = compose(使用 Epitath 解決嵌套問題, ) {(toggle, counter) => ( )}
Epitath 提供了一種新方式解決這個嵌套的問題:
const App = epitath(function*() { const { count } = yieldconst { on } = yield return ( ) })
renderProps 方案與 Epitath 方案,可以類比為 回調 方案與 async/await 方案。Epitath 和 compose 都解決了 renderProps 可能帶來的嵌套問題,而 compose 是通過將多個 renderProps merge 為一個,而 Epitath 的方案更接近 async/await 的思路,利用 generator 實現了偽同步代碼。
3 精讀Epitath 源碼一共 40 行,我們分析一下其精妙的方式。
下面是 Epitath 完整的源碼:
import React from "react"; import immutagen from "immutagen"; const compose = ({ next, value }) => next ? React.cloneElement(value, null, values => compose(next(values))) : value; export default Component => { const original = Component.prototype.render; const displayName = `EpitathContainer(${Component.displayName || "anonymous"})`; if (!original) { const generator = immutagen(Component); return Object.assign( function Epitath(props) { return compose(generator(props)); }, { displayName } ); } Component.prototype.render = function render() { // Since we are calling a new function to be called from here instead of // from a component class, we need to ensure that the render method is // invoked against `this`. We only need to do this binding and creation of // this function once, so we cache it by adding it as a property to this // new render method which avoids keeping the generator outside of this // method"s scope. if (!render.generator) { render.generator = immutagen(original.bind(this)); } return compose(render.generator(this.props)); }; return class EpitathContainer extends React.Component { static displayName = displayName; render() { returnimmutagen; } }; };
immutagen 是一個 immutable generator 輔助庫,每次調用 .next 都會生成一個新的引用,而不是自己發生 mutable 改變:
import immutagen from "immutagen"; const gen = immutagen(function*() { yield 1; yield 2; return 3; })(); // { value: 1, next: [function] } gen.next(); // { value: 2, next: [function] } gen.next(); // { value: 2, next: [function] } gen.next().next(); // { value: 3, next: undefined }compose
看到 compose 函數就基本明白其實現思路了:
const compose = ({ next, value }) => next ? React.cloneElement(value, null, values => compose(next(values))) : value;
const App = epitath(function*() { const { count } = yield; const { on } = yield ; });
通過 immutagen,依次調用 next,生成新組件,且下一個組件是上一個組件的子組件,因此會產生下面的效果:
yield yield yield// 等價于
到此其源碼精髓已經解析完了。
存在的問題crimx 在討論中提到,Epitath 方案存在的最大問題是,每次 render 都會生成全新的組件,這對內存是一種挑戰。
稍微解釋一下,無論是通過 原生的 renderProps 還是 compose,同一個組件實例只生成一次,React 內部會持久化這些組件實例。而 immutagen 在運行時每次執行渲染,都會生成不可變數據,也就是全新的引用,這會導致廢棄的引用存在大量 GC 壓力,同時 React 每次拿到的組件都是全新的,雖然功能相同。
4 總結epitath 巧妙的利用了 immutagen 的不可變 generator 的特性來生成組件,并且在遞歸 .next 時,將順序代碼解析為嵌套代碼,有效解決了 renderProps 嵌套問題。
喜歡 epitath 的同學趕快入手吧!同時我們也看到 generator 手動的步驟控制帶來的威力,這是 async/await 完全無法做到的。
是否可以利用 immutagen 解決 React Context 與組件相互嵌套問題呢?還有哪些其他前端功能可以利用 immutagen 簡化的呢?歡迎加入討論。
5 更多討論討論地址是:精讀《Epitath - renderProps 新用法》 · Issue #106 · dt-fe/weekly
如果你想參與討論,請點擊這里,每周都有新的主題,周末或周一發布。前端精讀 - 幫你篩選靠譜的內容。
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/98354.html
摘要:更容易將組件的與狀態分離。也就是只提供狀態處理方法,不會持久化狀態。大體思路是利用共享一份數據,作為的數據源。精讀帶來的約定函數必須以命名開頭,因為這樣才方便做檢查,防止用判斷包裹語句。前端精讀幫你篩選靠譜的內容。 1 引言 React Hooks 是 React 16.7.0-alpha 版本推出的新特性,想嘗試的同學安裝此版本即可。 React Hooks 要解決的問題是狀態共享,...
摘要:今天我們就來解讀一下的源碼。比較有意思,將定時器以方式提供出來,并且提供了方法。實現方式是,在組件內部維護一個定時器,實現了組件更新銷毀時的計時器更新銷毀操作,可以認為這種定時器的生命周期綁定了組件的生命周期,不用擔心銷毀和更新的問題。 1. 引言 React PowerPlug 是利用 render props 進行更好狀態管理的工具庫。 React 項目中,一般一個文件就是一個類,...
摘要:可以看到,這樣不僅沒有占用組件自己的,也不需要手寫回調函數進行處理,這些處理都壓縮成了一行。效果通過拿到周期才執行的回調函數。實現等價于的回調僅執行一次時,因此直接把回調函數拋出來即可。 1 引言 上周的 精讀《React Hooks》 已經實現了對 React Hooks 的基本認知,也許你也看了 React Hooks 基本實現剖析(就是數組),但理解實現原理就可以用好了嗎?學的是...
摘要:實際上是讓組件的接收函數,由函數來渲染內容。將通用的邏輯抽象在該組件的內部,然后依據業務邏輯來調用函數內渲染內容的函數,從而達到重用邏輯的目的。當然,上邊通過傳入了這屬于組件的增強功能。還有也提供了一種新模式來解決這個問題。 寫業務時,我們經常需要抽象一些使用頻率較高的邏輯,但是除了高階組件可以抽象邏輯,RenderProps也是一種比較好的方法。 RenderProps,顧名思義就是...
閱讀 2580·2021-11-24 09:38
閱讀 2612·2019-08-30 15:54
閱讀 926·2019-08-30 15:52
閱讀 1914·2019-08-30 15:44
閱讀 2721·2019-08-30 13:48
閱讀 776·2019-08-29 16:21
閱讀 1006·2019-08-29 14:03
閱讀 2221·2019-08-28 18:15