詳解React中共享組件邏輯的三種方式
廢話少說,這三種方式分別是:render props、高階組件和自定義Hook。下面依次演示
假設(shè)有一個TimeOnPage組件專門用來記錄用戶在當(dāng)前頁面停留時間,像這樣:
const TimeOnPage = () => {
const [second, setSecond] = useState(0);
useEffect(() => {
setTimeout(() => {
setSecond(second + 1);
}, 1000);
}, [second]);
return (
<div>停留時間:{second}秒</div>
);
}
如果另一個組件需要復(fù)用這個功能,我們能否封裝一下,以便輕松地與其它組件共享?
一般很自然地想到子組件嵌套的方式,利用props傳參
const Child = (props) => {
return <div>stayTime: {props.stayTime}s</div>;
};
const TimeOnPage = () => {
const [second, setSecond] = useState(0);
useEffect(() => {
setTimeout(() => {
setSecond(second + 1);
}, 1000);
}, [second]);
return (
<div>
<Child stayTime={second} />
</div>
);
}
這屬于在 TimeOnPage組件內(nèi)部硬編碼,還沒有達(dá)到封裝復(fù)用的目標(biāo)。看看render props怎么做?
render props
“render prop” 是指一種在 React 組件之間使用一個值為函數(shù)的 prop 共享代碼的簡單技術(shù)
接上文,在TimeOnPage里定義一個值為函數(shù)的prop,想渲染什么組件,在函數(shù)里返回即可,函數(shù)的參數(shù)就是想要共享的state。
const Child = (props) => {
return <div>stayTime: {props.stayTime}s</div>;
};
const TimeOnPage = (props) => {
const [second, setSecond] = useState(0);
useEffect(() => {
setTimeout(() => {
setSecond(second + 1);
}, 1000);
}, [second]);
return <div>{props.render(second)}</div>;
};
<TimeOnPage render={(stayTime) => <Child stayTime={stayTime} />
其實,render prop 就是一個用于告知組件需要渲染什么內(nèi)容的函數(shù)prop。
React Router也用到了這項技術(shù)。
<Router>
<Route path="/home" render={() => <div>Home</div>} />
</Router>
高階組件
高階組件(HOC)是 React 中用于復(fù)用組件邏輯的一種高級技巧。HOC 自身不是 React API 的一部分,它是一種基于 React 的組合特性而形成的設(shè)計模式。
高階組件是一個函數(shù),參數(shù)是一個需要被復(fù)用的組件A,返回值是一個新的組件N。新組件N是在組件A的基礎(chǔ)上做了一些加工,但不會修改組件A本身,只是功能增強(qiáng)。
假設(shè)有一個新聞列表組件長這樣:
const NewList = () => {
return (
<div>
<ul>
<li>news item</li>
<li>news item</li>
</ul>
</div>
);
}
想要在新聞列表加載期間顯示loading動畫組件 <Loading />,通常會這么做
const Loading = () => {
// loading動畫
}
const NewList = ({ isLoading }) => {
return isLoading ? (
<Loading />
) : (
<div>
<ul>
<li>news item</li>
<li>news item</li>
</ul>
</div>
);
};
假設(shè)現(xiàn)在Table組件也要在加載數(shù)據(jù)期間顯示loading動畫組件,遵循類似的模式
const Loading = () => {
// loading動畫
}
const DataList = ({ isLoading, ...props }) => {
return isLoading ? (
<Loading />
) : (
<Table {...props} />
);
};
以上,你會發(fā)現(xiàn)DataList和NewList結(jié)構(gòu)極度相似,如果還有第三個、第四個組件要加loading,繼續(xù)照這個模式重復(fù)第三次、第四次嗎?這不是最理想的做法,更好的做法是,使用高階組件把這個模式抽象出來:
const WithLoading = (WrappedComponent) => {
return ({isLoading, ...props}) => {
return isLoading ? <Loading /> : <WrappedComponent {...props} />;
}
};
然后就可以在不修改NewList和DataList的情況下分別給他們增加loading
const NewList = () => {
return (
<div>
<ul>
<li>news item</li>
<li>news item</li>
</ul>
</div>
);
};
const DataList = (props) => {
return <Table {...props} />
};
const WithLoading = (WrappedComponent) => {
return ({isLoading, ...props}) => {
return isLoading ? <Loading /> : <WrappedComponent {...props} />;
}
};
// 帶loading的NewList
const WithLoadingNewList = WithLoading(<NewList />)
// 帶loading的DataList
const WithLoadingDataList = WithLoading(<DataList />)
自定義Hook
Hook 是 React 16.8 的新增特性。它可以讓你在不編寫 class 的情況下使用 state 以及其他的 React 特性。
React Hook有useState、useEffect等,它們都是函數(shù),自定義Hook也是一個函數(shù),它的名稱同樣以use開頭,函數(shù)內(nèi)部可以調(diào)用其它Hook。與React組件不同的是,自定義Hook可以沒有返回值。與普通函數(shù)不同的是,自定義Hook內(nèi)部可以調(diào)用其它Hook,而普通函數(shù)則不行。
在寫業(yè)務(wù)邏輯過程中,一般會將一些可重用的的方法定義成工具函數(shù),然后就可以到處復(fù)用。同樣,通過自定義 Hook,可以將組件邏輯提取到可重用的函數(shù)中。到底選擇自定義Hook還是工具函數(shù),取決于要提取的組件邏輯需不需要用到其他Hook,如果需要,就選擇自定義Hook,否則用工具函數(shù)即可。
回到本文第一個 TimeOnPage組件,改成自定義Hook的形式
const useTimeOnPage = () => {
const [second, setSecond] = useState(0);
useEffect(() => {
setTimeout(() => {
setSecond(second + 1);
}, 1000);
}, [second]);
return second;
}
使用方法
const Demo = () => {
const stayTime = useTimeOnPage();
return <div>當(dāng)前頁面停留時間:{stayTime}秒</div>
}
總結(jié)
三種共享組件邏輯的方式有各自的適用場景:
render props適合共享那些有不同子組件/子元素的父組件,子組件/子元素的“坑位”已經(jīng)定義好了,只能渲染在指定位置;
高階組件適合在不修改原有組件的基礎(chǔ)上對組件進(jìn)行擴(kuò)展;
自定義Hook能做的,純函數(shù)基本上也能做,只是有時候用自定義Hook實現(xiàn)會更方便快捷。
本文鏈接:Github
到此這篇關(guān)于詳解React中共享組件邏輯的三種方式的文章就介紹到這了,更多相關(guān)React 共享組件邏輯內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- React組件重構(gòu)之嵌套+繼承及高階組件詳解
- React中嵌套組件與被嵌套組件的通信過程
- 編寫簡潔React組件的小技巧
- 使用hooks寫React組件需要注意的5個地方
- react實現(xiàn)Radio組件的示例代碼
- 詳解對于React結(jié)合Antd的Form組件實現(xiàn)登錄功能
- 基于react hooks,zarm組件庫配置開發(fā)h5表單頁面的實例代碼
- React antd tabs切換造成子組件重復(fù)刷新
- 在react-antd中彈出層form內(nèi)容傳遞給父組件的操作
- 在react項目中使用antd的form組件,動態(tài)設(shè)置input框的值
- 淺談React中組件邏輯復(fù)用的那些事兒
- React嵌套組件的構(gòu)建順序
相關(guān)文章
React中組件的this.state和setState的區(qū)別
在React開發(fā)中,this.state用于初始化和讀取組件狀態(tài),而setState()用于安全地更新狀態(tài),正確使用這兩者對于管理React組件狀態(tài)至關(guān)重要,避免性能問題和常見錯誤2024-09-09
詳解React Native網(wǎng)絡(luò)請求fetch簡單封裝
本篇文章主要介紹了詳解React Native網(wǎng)絡(luò)請求fetch簡單封裝,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-08-08
React實現(xiàn)控制減少useContext導(dǎo)致非必要的渲染詳解
這篇文章主要介紹了React如何有效減少使用useContext導(dǎo)致的不必要渲染,使用useContext在改變一個數(shù)據(jù)時,是通過自己逐級查找對比改變的數(shù)據(jù)然后渲染,本文通過示例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2022-11-11
React Native設(shè)備信息查看調(diào)試詳解
這篇文章主要為大家介紹了React Native設(shè)備信息查看調(diào)試詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-11-11
React根據(jù)當(dāng)前頁面路由進(jìn)行自動高亮示例代碼
要根據(jù)當(dāng)前頁面路由自動高亮頂部菜單項,可以使用 React Router 的 useLocation 鉤子來獲取當(dāng)前路徑,并根據(jù)路徑動態(tài)設(shè)置菜單項的高亮效果,本文給大家介紹了一個完整的示例,展示如何根據(jù)當(dāng)前頁面路由自動高亮頂部菜單項,需要的朋友可以參考下2024-07-07
react中useEffect函數(shù)的詳細(xì)用法(最新推薦)
useEffect是React中的一個Hook,用于在函數(shù)組件中處理副作用(如數(shù)據(jù)獲取、訂閱、手動更改 DOM 等),useEffect屬于組件的生命周期方法,下面通過本文給大家分享react中useEffect函數(shù)的詳細(xì)用法,感興趣的朋友跟隨小編一起看看吧2024-06-06

