React-redux?中useSelector使用源碼分析
在一個 action 被分發(fā)(dispatch) 后,useSelector() 默認對 select 函數(shù)的返回值進行引用比較 ===,并且僅在返回值改變時觸發(fā)重渲染。但是,不同于 connect(),useSelector()并不會阻止父組件重渲染導(dǎo)致的子組件重渲染的行為,即使組件的 props 沒有發(fā)生改變。
useSelector 源碼分析
import { useContext, useEffect, useReducer, useRef, } from 'react'; import StoreContext from './context'; type EqualityFn = (a: T, b: T) => boolean; export default function useSelector<T, Selected extends unknown>( selector: (state: T) => Selected, equalityFn?: EqualityFn, ): Selected { const store = useContext(StoreContext); // 注:react-redux@8-beta 中使用 React18提供的useSyncExternalStoreapi 來做強制更新。 // 17版本之前采用下面的方式進行強制render const [, forceRender] = useReducer((s) => s + 1, 0); // 保存最近一次從store中獲取的state const latestStoreState = useRef(store.getState()); // 保存最近一次通過selector函數(shù)返回的值 - 也就是useSelector 返回的值 const latestSelectedState = useRef(selector(latestStoreState.current)); useEffect(() => { // checkUpdate - 判斷是否強制更新; function checkUpdate() { const newState = store.getState(); // state 沒有改變,直接返回;發(fā)生在 reducer 中default的情況; // 其他情況使用immer,都會返回一個新的state對象(本質(zhì)還是淺拷貝) if (newState === latestStoreState) return; const newSelectedState = selector(newState); // 默認的 equalityFn 進行的是 === 判斷; // 所以默認的equalityFn函數(shù)的 === 決定了, // 1 我們不能通過selector函數(shù)去返回創(chuàng)建一個新的對象進行返回 // 2 利用淺拷貝的特性,盡量返回state上最小的粒度,保證當(dāng)前selector返回的值沒有改變時不會執(zhí)行強制更新 if (!equalityFn) equalityFn = (a, b) => a === b; // 對比新舊SelectedState 數(shù)據(jù) 是否全等,如果不等 則進行強制更新 if (!equalityFn(newSelectedState, latestSelectedState.current)) { latestSelectedState.current = newSelectedState; latestStoreState.current = newState; forceRender(); } } // 執(zhí)行dispatch({type:XXX}) ,會調(diào)用 store.subscribe進行監(jiān)聽 // 執(zhí)行checkUpdate函數(shù) const unsubscribe = store.subscribe(checkUpdate); return () => unsubscribe(); }, [store]); // useSelector 返回最新的seletor函數(shù)返回值 return latestSelectedState.current; } 不同引用的探究 // 寫法一 const { project,pages } = useSelector((state: IRootState) => state.prototype); // 寫法二 const project = useSelector((state: IRootState) => state.prototype.project); const project = useSelector((state: IRootState) => state.prototype.project); // 寫法三 (傳遞一個equalityFn 函數(shù)進行判斷 ) const { project,pages } = useSelector((state: IRootState) => state.prototype,equalityFn);
結(jié)論
1.寫法一,不可取。因為reducer 源碼中,會緩存上一次返回的oldSelectdState,和新獲取的selectdState 進行全等判斷。如果直接返回一個模塊,一旦這個模塊中任意值發(fā)生變化,整個模塊值都會改變,即使我們使用到的project 和 pages 數(shù)據(jù)沒有發(fā)生改變,也會讓該組件執(zhí)行重新更新。
2.寫法二,可取。因為每一個useSelector 的 seletor注冊函數(shù)返回值都是模塊內(nèi)最小粒度。如果這個值沒有發(fā)生改變,就不會執(zhí)行更新。多個useSelector 如果其中有多處需要更新的話,react中會進行批量更新,也只會強制更新一次,對性能不會有影響。
3.寫法三,如果想要返回一個對象,可以傳遞一個equalityFn 進行深度比較,不采用 === 比較方式。
缺點是,如果數(shù)據(jù)量很大或者嵌套很深,深度比較會有性能問題。
到此這篇關(guān)于React-redux 中useSelector使用的文章就介紹到這了,更多相關(guān)React-redux useSelector使用內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
react-native 實現(xiàn)購物車滑動刪除效果的示例代碼
這篇文章主要介紹了react-native 實現(xiàn)購物車滑動刪除效果的示例代碼,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-01-01React實現(xiàn)文件上傳和斷點續(xù)傳功能的示例代碼
這篇文章主要為大家詳細介紹了React實現(xiàn)文件上傳和斷點續(xù)傳功能的相關(guān)知識,文中的示例代碼講解詳細,感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2024-02-02React使用setState更新數(shù)組的方法示例(追加新數(shù)據(jù))
在?React?中,setState?是管理組件狀態(tài)的核心方法之一,然而,當(dāng)我們需要更新狀態(tài)中的數(shù)組時,如何高效且安全地操作變得尤為關(guān)鍵,本文將詳細解析以下代碼的實現(xiàn)邏輯,幫助你掌握在?React?中追加數(shù)組數(shù)據(jù)的最佳實踐,需要的朋友可以參考下2025-03-03react實現(xiàn)動態(tài)增減表單項的示例代碼
在做項目的時候,甲方給的信息有限,網(wǎng)頁的備案信息寫成固定的,之后驗收的時候,甲方要求把這個備案信息寫成動態(tài)的,可以自增減,下面通過實例代碼給大家介紹react實現(xiàn)動態(tài)增減表單項的示例,感興趣的朋友跟隨小編一起看看吧2024-05-05