一文帶你搞懂React中的useReducer
useReducer 是除useState之外另一個(gè)與狀態(tài)管理相關(guān)的 hook,對(duì)于熟悉 Redux 的工程師而言,理解 useReducer 將變得簡單,在 React 內(nèi)部,useState 由 useReducer 實(shí)現(xiàn)。
useReducer 的類型定義
useReducer 的類型定義如下:
function useReducer<R extends ReducerWithoutAction<any>, I>( reducer: R, initializerArg: I, initializer: (arg: I) => ReducerStateWithoutAction<R> ): [ReducerStateWithoutAction<R>, DispatchWithoutAction]; function useReducer<R extends ReducerWithoutAction<any>>( reducer: R, initializerArg: ReducerStateWithoutAction<R>, initializer?: undefined ): [ReducerStateWithoutAction<R>, DispatchWithoutAction]; function useReducer<R extends Reducer<any, any>, I>( reducer: R, initializerArg: I & ReducerState<R>, initializer: (arg: I & ReducerState<R>) => ReducerState<R> ): [ReducerState<R>, Dispatch<ReducerAction<R>>]; function useReducer<R extends Reducer<any, any>, I>( reducer: R, initializerArg: I, initializer: (arg: I) => ReducerState<R> ): [ReducerState<R>, Dispatch<ReducerAction<R>>]; function useReducer<R extends Reducer<any, any>>( reducer: R, initialState: ReducerState<R>, initializer?: undefined ): [ReducerState<R>, Dispatch<ReducerAction<R>>];
useReducer 的類型定義很復(fù)雜,一共有 5 個(gè)重載,總體而言,它最多接受3個(gè)參數(shù),第 1 個(gè)參數(shù)是一個(gè)用于更新狀態(tài)的函數(shù),之后將它稱為 reducer。第 3 個(gè)參數(shù)非必填,如果不存在第 3 個(gè)參數(shù),那么第 2 個(gè)參數(shù)將作為狀態(tài)的初始值;如果存在第 3 個(gè)參數(shù),那么它必須是函數(shù),此時(shí)第 2 個(gè)參數(shù)被傳遞給該函數(shù)用于計(jì)算狀態(tài)的初始值,該函數(shù)只在組件初始渲染時(shí)執(zhí)行一次。useReducer 的返回值是一個(gè)長度為 2 的數(shù)組,數(shù)組的第 1 個(gè)位置是狀態(tài)值,第 2 個(gè)位置是一個(gè)用于觸發(fā)狀態(tài)更新的函數(shù),將它記為dispatch,調(diào)用 dispatch 將導(dǎo)致 reducer 被調(diào)用。接下來通過計(jì)數(shù)器 demo 對(duì)比 useState 和useReducer 用法上的差異。
用 useState 實(shí)現(xiàn)計(jì)數(shù)器
用 useState 實(shí)現(xiàn)計(jì)數(shù)器,代碼如下:
function UseStateCounterDemo() { const [value, setValue] = useState<number>(0) const [step, setStep] = useState<number>(1) const onChangeStep = (ev: React.ChangeEvent<HTMLInputElement>) => { setStep(Number(ev.target.value)) } return ( <div> <h2>用useState實(shí)現(xiàn)計(jì)數(shù)器</h2> count: {count}; step: <input type='number' value={step} onChange={onChangeStep}/> <button onClick={() => setValue(value + step)}>加</button> <button onClick={() => setValue(value - step)}>減</button> <button onClick={() => {setValue(0); setStep(1)}}>重置</button> </div> ) }
上述代碼很簡單,它使用 useState 定義了兩個(gè)狀態(tài),分別表示計(jì)數(shù)器的值和加減步數(shù),在過去的文章里已對(duì) useState 做過詳細(xì)的介紹,這里不再贅述,下面詳細(xì)介紹用 useReducer 實(shí)現(xiàn)計(jì)數(shù)器。
用 useReducer 實(shí)現(xiàn)計(jì)數(shù)器
定義 reducer
使用 useReducer hook 離不開reducer,它是一個(gè)函數(shù),用于更新 useReducer 返回的狀態(tài),代碼如下:
const initArg: Counter = {value: 0, step: 1} // 它用于更新state function reducer(prevState: Counter, action: Action): Counter { switch (action.type) { case 'increment': return { ...prevState, value: prevState.value + prevState.step } case 'decrement': return { ...prevState, value: prevState.value - prevState.step } case 'reset': return initArg case 'changeStep': return { ...prevState, step: action.value || initArg.value } default: throw new Error(); } }
reducer 用于更新狀態(tài),計(jì)數(shù)器 demo 有兩個(gè)狀態(tài),分別是計(jì)數(shù)器的當(dāng)前值和它的加減步數(shù),這兩個(gè)狀態(tài)密切相關(guān),這里用一個(gè) TS 接口去描述它,代碼如下:
interface Counter { // 計(jì)數(shù)器的當(dāng)前值 value: number // 計(jì)數(shù)器的加減步數(shù) step: number }
計(jì)數(shù)器有 3 種操作,分別是加、減、重置和修改步數(shù),這里用 TS 接口描述這些行為,代碼如下:
interface Action { type: 'increment' | 'decrement' | 'reset'|'changeStep', value?: number }
在組件中使用 useReducer
// 計(jì)數(shù)器的初始值 const initArg: Counter = {value: 0, step: 1} function UseReducerCounterDemo() { // 使用上一步定義 reducer const [counter, dispatch] = useReducer(reducer, initArg) const onChangeStep = (ev: React.ChangeEvent<HTMLInputElement>) => { dispatch({type: 'changeStep',value: Number(ev.target.value)}) } return ( <div> <h2>用useReducer實(shí)現(xiàn)計(jì)數(shù)器</h2> count: {counter.value}; step: <input type='number' value={counter.step} onChange={onChangeStep}/> <button onClick={() => dispatch({type: 'increment'})}>加</button> <button onClick={() => dispatch({type: 'decrement'})}>減</button> <button onClick={() => dispatch({type: 'reset'})}>重置</button> </div> ) }
只考慮代碼量,讀者應(yīng)該都會(huì)認(rèn)為useState比useReducer更簡潔。仔細(xì)觀察可以發(fā)現(xiàn),上述計(jì)數(shù)器除了有value還有step,step對(duì)value有影響,UseStateCounterDemo組件將它們零散地保存在不同的狀態(tài)中,UseStateCounterDemo組件將它們關(guān)聯(lián)在同一個(gè)狀態(tài)中,內(nèi)聚性更高。useState與useReducer沒有優(yōu)劣之外,它們有各自適用的場景,這里有如下建議:
1.當(dāng)狀態(tài)是一個(gè)擁有很多屬性的復(fù)雜對(duì)象,并且狀態(tài)更新涉及復(fù)雜的邏輯時(shí),推薦使用useReducer。
2.當(dāng)某個(gè)狀態(tài)的更新受另一個(gè)狀態(tài)影響時(shí),推薦使用 useReducer將它們放在一起。
3.當(dāng)狀態(tài)只是單獨(dú)的基本數(shù)據(jù)類型時(shí),推薦使用 useState。
在介紹 useEffect時(shí)曾強(qiáng)調(diào),為了在 effect 中拿到狀態(tài)最新的值,必須給 effect 設(shè)置正確地依賴項(xiàng)。在 useEffect 中使用 useReducer 返回的 dispatch 能讓 effect 自給自足,減少依賴項(xiàng)。示例代碼如下:
function DispatchDemo(props: {step: number}) { function reducer(value: number) { // 始終能訪問到最新的 step return props.step + value } const [value, dispatch] = useReducer(reducer, 0) useEffect(() => { document.body.addEventListener('click', dispatch) return () => { document.body.removeEventListener('click', dispatch) } }, []) return // todo }
上述代碼 useEffect 的第二個(gè)參數(shù)為空數(shù)組,這意味著 effect 只在組件初始渲染時(shí)執(zhí)行。由于React 會(huì)讓 dispatch 在組件的每次渲染中保持唯一的引用,所以 dispatch 不必出現(xiàn)在effect的依賴中,此特性與 ref 類似。雖然 dispatch 的引用保持不變,但它能調(diào)用組件本次渲染時(shí)的reducer,在 reducer 作用域?qū)⒌玫阶钚碌?state 和 props。
推薦閱讀:
我搞懂了 React 的 useState 和 useEffect
到此這篇關(guān)于一文帶你搞懂React中的useReducer的文章就介紹到這了,更多相關(guān)React useReducer內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
詳解create-react-app 2.0版本如何啟用裝飾器語法
這篇文章主要介紹了詳解create-react-app 2.0版本如何啟用裝飾器語法,cra2.0時(shí)代如何啟用裝飾器語法呢? 我們依舊采用的是react-app-rewired, 通過劫持webpack cofig對(duì)象, 達(dá)到修改的目的2018-10-10基于React實(shí)現(xiàn)倒計(jì)時(shí)功能
這篇文章主要為大家詳細(xì)介紹了如何基于React實(shí)現(xiàn)倒計(jì)時(shí)功能,文中的示例代碼講解詳細(xì),具有一定的借鑒價(jià)值,有需要的小伙伴可以參考一下2024-02-02React?Hooks之usePolymerAction抽象代碼結(jié)構(gòu)設(shè)計(jì)理念
這篇文章主要為大家介紹了React?Hooks之usePolymerAction抽象代碼結(jié)構(gòu)設(shè)計(jì)理念,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-09-09React中如何設(shè)置多個(gè)className
這篇文章主要介紹了React中如何設(shè)置多個(gè)className問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-01-01手挽手帶你學(xué)React之React-router4.x的使用
這篇文章主要介紹了手挽手帶你學(xué)React之React-router4.x的使用,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-02-02React 添加引用路徑時(shí)如何使用@符號(hào)作為src文件
這篇文章主要介紹了React 添加引用路徑時(shí)如何使用@符號(hào)作為src文件,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-06-06