React中useEffect原理的代碼簡(jiǎn)單實(shí)現(xiàn)詳解
React的useEffect
鉤子是React函數(shù)組件中處理副作用(例如API請(qǐng)求、訂閱或手動(dòng)修改DOM等)的重要工具。在本文中,將通過(guò)一個(gè)簡(jiǎn)單的例子解釋如何用代碼實(shí)現(xiàn)useEffect
的基本原理。
首先,定義了兩個(gè)全局變量,用于跟蹤不同的副作用狀態(tài):
let prevDepsAry = []; // 存放依賴項(xiàng)數(shù)組的上一次值 let effectIndex = 0; // 當(dāng)前副作用索引
prevDepsAry
用于記錄上次渲染時(shí)的依賴項(xiàng)數(shù)組,effectIndex
用來(lái)標(biāo)示當(dāng)前處理的副作用的位置。
接下來(lái)是useEffect
函數(shù)的實(shí)現(xiàn):
function useEffect(callback, depsAry) { // 校驗(yàn)callback是否為函數(shù) if (Object.prototype.toString.call(callback) !== '[object Function]') { throw new Error('useEffect的第一個(gè)參數(shù)必須是一個(gè)函數(shù)'); } // 不提供依賴項(xiàng)數(shù)組的情況下,默認(rèn)每次渲染都執(zhí)行callback if (typeof depsAry === 'undefined') { callback(); } else { // 校驗(yàn)depsAry是否為數(shù)組 if (Object.prototype.toString.call(depsAry) !== '[object Array]') { throw new Error('useEffect的第二個(gè)參數(shù)必須是數(shù)組'); } // 獲取依賴項(xiàng)數(shù)組的前一個(gè)值 let prevDeps = prevDepsAry[effectIndex]; // 比較依賴項(xiàng)數(shù)組的每一個(gè)值,確定是否發(fā)生變化 let hasChanged = prevDeps ? !depsAry.every((dep, index) => dep === prevDeps[index]) : true; // 如果依賴項(xiàng)變化或者是首次渲染(hasChanged為true),則執(zhí)行callback if (hasChanged) { callback(); } // 將當(dāng)前的依賴項(xiàng)數(shù)組存儲(chǔ)起來(lái),用于下次渲染時(shí)比較 prevDepsAry[effectIndex] = depsAry; } // 增加副作用索引,準(zhǔn)備下一個(gè)副作用 effectIndex++; }
useEffect
的實(shí)現(xiàn)邏輯為:
- 驗(yàn)證傳入的
callback
是否為函數(shù)。如果不是,拋出錯(cuò)誤。 - 如果沒(méi)有傳入
depsAry
,那么每次組件渲染時(shí)都執(zhí)行callback
。 - 如果傳入了
depsAry
,首先驗(yàn)證其為數(shù)組。然后,獲取上一次的依賴數(shù)組并與當(dāng)前數(shù)組逐項(xiàng)比較。如果存在差異或者是首次渲染,則執(zhí)行callback
。 - 更新
prevDepsAry
的對(duì)應(yīng)項(xiàng),在下次組件渲染時(shí)用作比較。 effectIndex
自增,確保下一個(gè)useEffect
的處理使用正確的索引。
下一步定義了render
函數(shù):
function render() { // 重設(shè)副作用索引 effectIndex = 0; // 使用ReactDOM來(lái)渲染App組件 ReactDOM.render(<App />, document.getElementById('root')); }
在每次渲染前,effectIndex
必須重置為0,這保證了useEffect
在處理依賴項(xiàng)時(shí)的正確性。
最后,在App組件中使用useEffect
:
function App() { // 使用自定義的useState和useEffect useEffect(() => { console.log('副作用函數(shù)執(zhí)行了'); // 在這里可能會(huì)執(zhí)行如API請(qǐng)求、訂閱事件等具有副作用的操作 // 當(dāng)依賴項(xiàng)發(fā)生變化時(shí),這里的代碼會(huì)被執(zhí)行 }, [/* 依賴項(xiàng)數(shù)組 */]); return ( // 組件的JSX結(jié)構(gòu) <div></div> ); }
在組件中通過(guò)調(diào)用useEffect
,并傳遞一個(gè)執(zhí)行副作用操作的函數(shù)以及依賴項(xiàng)數(shù)組,實(shí)現(xiàn)了依賴項(xiàng)變化時(shí)執(zhí)行副作用邏輯的需求。
最后一步,調(diào)用render()
以觸發(fā)首次渲染。
通過(guò)上述代碼,我單地回答了如何實(shí)現(xiàn)useEffect
的問(wèn)題。當(dāng)然,實(shí)際的ReactuseEffect
實(shí)現(xiàn)更加復(fù)雜,并且涉及到調(diào)度和清理操作,但上面的代碼為理解其基本原理提供了良好的起點(diǎn)。
到此這篇關(guān)于React中useEffect原理的代碼簡(jiǎn)單實(shí)現(xiàn)詳解的文章就介紹到這了,更多相關(guān)React useEffect內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
淺析JS中什么是自定義react數(shù)據(jù)驗(yàn)證組件
我們?cè)谧銮岸吮韱翁峤粫r(shí),經(jīng)常會(huì)遇到要對(duì)表單中的數(shù)據(jù)進(jìn)行校驗(yàn)的問(wèn)題。這篇文章主要介紹了js中什么是自定義react數(shù)據(jù)驗(yàn)證組件,需要的朋友可以參考下2018-10-10如何對(duì)react hooks進(jìn)行單元測(cè)試的方法
這篇文章主要介紹了如何對(duì)react hooks進(jìn)行單元測(cè)試的方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-08-08使用React18和WebSocket構(gòu)建實(shí)時(shí)通信功能詳解
WebSocket是一種在Web應(yīng)用中實(shí)現(xiàn)雙向通信的協(xié)議,它允許服務(wù)器主動(dòng)向客戶端推送數(shù)據(jù),而不需要客戶端發(fā)起請(qǐng)求,本文將探索如何在React?18應(yīng)用中使用WebSocket來(lái)實(shí)現(xiàn)實(shí)時(shí)通信,感興趣的可以了解下2024-01-01聊聊ant?design?charts?獲取后端接口數(shù)據(jù)展示問(wèn)題
今天在做項(xiàng)目的時(shí)候遇到幾個(gè)讓我很頭疼的問(wèn)題,一個(gè)是通過(guò)后端接口成功訪問(wèn)并又返回?cái)?shù)據(jù),但拿不到數(shù)據(jù)值。其二是直接修改state中的data,console中數(shù)組發(fā)生變化但任然數(shù)據(jù)未顯示,這篇文章主要介紹了ant?design?charts?獲取后端接口數(shù)據(jù)展示,需要的朋友可以參考下2022-05-05詳解React中父子組件數(shù)據(jù)傳遞和修改的方式及原理
這篇文章主要為大家詳細(xì)介紹了React中父子組件數(shù)據(jù)傳遞和修改的方式及原理,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2024-04-04