React-redux?中useSelector使用源碼分析
在一個 action 被分發(fā)(dispatch) 后,useSelector() 默認對 select 函數的返回值進行引用比較 ===,并且僅在返回值改變時觸發(fā)重渲染。但是,不同于 connect(),useSelector()并不會阻止父組件重渲染導致的子組件重渲染的行為,即使組件的 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函數返回的值 - 也就是useSelector 返回的值
const latestSelectedState = useRef(selector(latestStoreState.current));
useEffect(() => {
// checkUpdate - 判斷是否強制更新;
function checkUpdate() {
const newState = store.getState();
// state 沒有改變,直接返回;發(fā)生在 reducer 中default的情況;
// 其他情況使用immer,都會返回一個新的state對象(本質還是淺拷貝)
if (newState === latestStoreState) return;
const newSelectedState = selector(newState);
// 默認的 equalityFn 進行的是 === 判斷;
// 所以默認的equalityFn函數的 === 決定了,
// 1 我們不能通過selector函數去返回創(chuàng)建一個新的對象進行返回
// 2 利用淺拷貝的特性,盡量返回state上最小的粒度,保證當前selector返回的值沒有改變時不會執(zhí)行強制更新
if (!equalityFn) equalityFn = (a, b) => a === b;
// 對比新舊SelectedState 數據 是否全等,如果不等 則進行強制更新
if (!equalityFn(newSelectedState, latestSelectedState.current)) {
latestSelectedState.current = newSelectedState;
latestStoreState.current = newState;
forceRender();
}
}
// 執(zhí)行dispatch({type:XXX}) ,會調用 store.subscribe進行監(jiān)聽
// 執(zhí)行checkUpdate函數
const unsubscribe = store.subscribe(checkUpdate);
return () => unsubscribe();
}, [store]);
// useSelector 返回最新的seletor函數返回值
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 函數進行判斷 )
const { project,pages } = useSelector((state: IRootState) => state.prototype,equalityFn);
結論
1.寫法一,不可取。因為reducer 源碼中,會緩存上一次返回的oldSelectdState,和新獲取的selectdState 進行全等判斷。如果直接返回一個模塊,一旦這個模塊中任意值發(fā)生變化,整個模塊值都會改變,即使我們使用到的project 和 pages 數據沒有發(fā)生改變,也會讓該組件執(zhí)行重新更新。
2.寫法二,可取。因為每一個useSelector 的 seletor注冊函數返回值都是模塊內最小粒度。如果這個值沒有發(fā)生改變,就不會執(zhí)行更新。多個useSelector 如果其中有多處需要更新的話,react中會進行批量更新,也只會強制更新一次,對性能不會有影響。
3.寫法三,如果想要返回一個對象,可以傳遞一個equalityFn 進行深度比較,不采用 === 比較方式。
缺點是,如果數據量很大或者嵌套很深,深度比較會有性能問題。
到此這篇關于React-redux 中useSelector使用的文章就介紹到這了,更多相關React-redux useSelector使用內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
React使用setState更新數組的方法示例(追加新數據)
在?React?中,setState?是管理組件狀態(tài)的核心方法之一,然而,當我們需要更新狀態(tài)中的數組時,如何高效且安全地操作變得尤為關鍵,本文將詳細解析以下代碼的實現邏輯,幫助你掌握在?React?中追加數組數據的最佳實踐,需要的朋友可以參考下2025-03-03

