欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

React使用Context的一些優(yōu)化建議

 更新時(shí)間:2024年04月10日 08:57:57   作者:奇舞精選  
Context?提供了一個(gè)無需為每層組件手動(dòng)添加?props,就能在組件樹間進(jìn)行數(shù)據(jù)傳遞的方法,本文為大家整理了React使用Context的一些優(yōu)化建議,希望對(duì)大家有所幫助

常用 API

React.createContext

const MyContext = React.createContext(defaultValue);

創(chuàng)建一個(gè) Context 對(duì)象。當(dāng) React 渲染一個(gè)訂閱了這個(gè) Context 對(duì)象的組件,這個(gè)組件會(huì)從組件樹中離自身最近的那個(gè)匹配的 Provider 中讀取到當(dāng)前的 context 值。

Context.Provider

<MyContext.Provider value={/* 某個(gè)值 */}>

每個(gè) Context 對(duì)象都會(huì)返回一個(gè) Provider React 組件,它允許消費(fèi)組件訂閱 context 的變化。

Provider 接收一個(gè) value 屬性,傳遞給消費(fèi)組件。一個(gè) Provider 可以和多個(gè)消費(fèi)組件有對(duì)應(yīng)關(guān)系。多個(gè) Provider 也可以嵌套使用,里層的會(huì)覆蓋外層的數(shù)據(jù)。

useContext

const store = useContext(MyContext)

接收一個(gè) context 對(duì)象(React.createContext 的返回值)并返回該 context 的當(dāng)前值。當(dāng)前的 context 值由上層組件中距離當(dāng)前組件最近的 <MyContext.Provider> 的 value prop 決定。

當(dāng) Provider 的 value 值發(fā)生變化時(shí),它內(nèi)部的所有消費(fèi)組件都會(huì)重新渲染

了解了 API 后,我們來看一個(gè)簡單的例子。

示例

index.js

const MyContext = React.createContext(null);

function reducer(state, action) {
  switch (action.type) {
    case 'addCount': {
      return {
        ...state,
        count: state.count + 1
      }
    }
    case 'addNum': {
      return {
        ...state,
        num: state.num + 1
      }
    }
    default: return state;
  }
}

const MyProvider = ({ children }) => {
  const [store, dispatch] = useReducer(reducer, { count: 0, num: 0 })
  return <MyContext.Provider value={{store, dispatch}}>{children}</MyContext.Provider>
};

export default () => {
  return (
    <MyProvider>
      <ChildCount />
      <ChildNum />
      <Child  />
    </MyProvider>
  );
}

ChildCount.js

export default () => {
    const { state, dispatch } = React.useContext(MyContext);
    console.log('re-render ChildCount', state.count)
    return (
      <>
        <div>count is: {state.count}</div>
        <button onClick={() => dispatch({ type: 'addCount' })}> AddCount </button>
      </>
    )
}

ChildNum.js

export default () => {
    const { state, dispatch } = React.useContext(MyContext);
    console.log('re-render ChildNum', state.num)
    return (
      <>
        <div>num is: {state.num}</div>
        <button onClick={() => dispatch({ type: 'addNum' })}> AddNum </button>
      </>
    )
}

Child.js

export default () => {
  console.log('re-render Child')
  return <div>Child</div>
}

點(diǎn)擊 AddCount 按鈕,輸出:

re-render ChildCount 1re-render ChildNum 0

點(diǎn)擊 AddNum 按鈕,輸出:

re-render ChildCount 1re-render ChildNum 1

我們可以發(fā)現(xiàn),Context.Provider 下的所有消費(fèi)組件,在 Provider.value 變化后,都會(huì) re-render

改變 count 、num 任意一個(gè)值,ChildCount,ChildNum 都會(huì) re-render

針對(duì)以上 re-render 情況,有以下方案可以優(yōu)化

優(yōu)化

針對(duì)子組件做函數(shù)記憶

React.memo

我們?nèi)缦滦薷乃械?Child 組件

export default React.memo(() => {
    const { state, dispatch } = React.useContext(MyContext);
    console.log('re-render ChildCount', state.count)
    return (
      <>
        <div>count is: {state.count}</div>
        <button onClick={() => dispatch({ type: 'addCount' })}> AddCount </button>
      </>
    )
})

點(diǎn)擊 AddCount 后發(fā)現(xiàn),依然打印出

re-render ChildCount 1
re-render ChildNum 0

我們重新認(rèn)識(shí)下 React.memo

React.memo 默認(rèn)情況下僅僅對(duì)傳入的 props 做淺比較,如果是內(nèi)部自身狀態(tài)更新(useState, useContext等),依然會(huì)重新渲染,在上面的例子中,useContext 返回的 state 一直在變化,導(dǎo)致就算被 memo 包裹的組件依然觸發(fā)更新了。

useMemo

我們?nèi)缦滦薷乃械?Child 組件

export default () => {
    const { state, dispatch } = React.useContext(MyContext);
    return useMemo(() => {
      console.log('re-render ChildCount', state.count)
      return (
          <>
            <div>count is: {state.count}</div>
            <button onClick={() => dispatch({ type: 'addCount' })}> AddCount </button>
          </>
      )
    }, [state.count, dispatch])
}

點(diǎn)擊 addCount 后發(fā)現(xiàn),只打印出了

re-render ChildCount 1

點(diǎn)擊 addNum 后發(fā)現(xiàn),只打印出了

re-render ChildNum 1

useMemo 可以做更細(xì)粒度的緩存,我們可以在依賴數(shù)組里來管理組件是否更新

我們可以思考一下,有沒有一種辦法,不用 useMemo 也可以做到按需渲染。就像 react-redux 中 useSelector 一樣實(shí)現(xiàn)按需渲染

動(dòng)手實(shí)現(xiàn) useSelector

我們先想一下,在上面的例子中,觸發(fā)子組件re-render的原因是什么?

沒錯(cuò)就是因?yàn)?Provider.value 的值一直在變更,那我們要想個(gè)辦法讓子組件感知不到 value 的變更,同時(shí)在 value 的某個(gè)值發(fā)生變更的時(shí)候,能夠觸發(fā)消費(fèi) value 的子組件 re-render

我們使用 觀察者模式 實(shí)現(xiàn)

1、我們使用 useMemo 緩存首次的 value,讓子組件感知不到 value 的變化

2、如果 value 不變化,那子組件就不會(huì)re-render,此時(shí)我們需要在真正 value 變化的時(shí)候,re-render 子組件,我們需要一個(gè) hooks(useSelector) 幫助我們實(shí)現(xiàn)子組件 re-render

3、子組件在初始化時(shí),useSelector 要幫助其訂閱 state 變更的回調(diào)函數(shù),并返回最新的 state(函數(shù)內(nèi)部獲取前后兩次的 state 做對(duì)比,不一樣則強(qiáng)制更新組件)

4、在 Context.Provider 中創(chuàng)建一個(gè)收集子組件訂閱state變更回調(diào)的集合,在其內(nèi)部監(jiān)聽 state(value),如果變更則遍歷集合,執(zhí)行所有回調(diào)函數(shù)

基于以上,我們依次實(shí)現(xiàn)了Context.Provider, useSelector, useDispatch

Context.Provider

const MyProvider = ({children}) => {
  const [state, dispatch] = useReducer(reducer, initState);
  
  // ref state
  const stateRef = useRef(null);
  stateRef.current = state;

  // ref 訂閱回調(diào)數(shù)組
  const subscribersRef = useRef([]);

  // state 變化,遍歷執(zhí)行回調(diào)
  useEffect(() => {
    subscribersRef.current.forEach(sub => sub());
  }, [state]);

  // 緩存 value, 利用 ref 拿到最新的 state, subscribe 狀態(tài)
  const value = useMemo(
    () => ({
      dispatch,
      subscribe: cb => {
        subscribersRef.current.push(cb);
        return () => {
          subscribersRef.current = subscribersRef.current.filter(item => item !== cb);
        };
      },
      getState: () => stateRef.current
    }),
    []
  )

  return <MyContext.Provider children={children} value={value} />;
}

useSelector

export const useSelector = selector => {
  // 強(qiáng)制更新
  const [, forceRender] = useReducer(v => v + 1, 0);
  const store = useContext(MyContext);

  // 獲取當(dāng)前使用的 state
  const selectedStateRef = useRef(null)
  selectedStateRef.current = selector(store.getState());

  // 對(duì)比更新回調(diào)
  const checkForUpdates = useCallback(() => {
    // 獲取變更后的 state
    const newState = selector(store.getState());
    // 對(duì)比前后兩次 state
    if (newState !== selectedStateRef.current) forceRender({});
  }, [store]);
  
  // 訂閱 state
  useEffect(() => {
    const subscription = store.subscribe(checkForUpdates);
    return () => subscription();
  }, [store, checkForUpdates]);
  
  // 返回需要的 state
  return selectedStateRef.current;
}

useDispatch

export const useDispatch = () => {
  const store = useContext(MyContext);
  return store.dispatch
}

我們用上面重寫的 API,改寫下剛開始的例子

index.js

export default () => {
    return (
      <MyProvider>
        <ChildCount />
        <ChildNum />
        <Child />
      </Provider>
    );
}

ChildCount.js

export default () => {
    const dispatch = useDispatch();
    const count = useSelector(state => state.count);
    console.log('re-render ChildCount', count)
    return (
      <>
        <div>count is: {count}</div>
        <button onClick={() => dispatch({ type: 'addCount' });}> AddCount </button>
      </>
    )
};

ChildNum.js

export default () => {
    const dispatch = useDispatch();
    const num = useSelector(state => state.num);
    console.log('re-render ChildNum', num)
    return (
      <>
        <div>num is: {num}</div>
        <button onClick={() => dispatch({ type: 'addNum' });}> AddNum </button>
      </>
    )
}

Child.js

export default () => {
    console.log('re-render Child')
    return <div>Child</div>
}

點(diǎn)擊AddCount: 只打印了 re-render ChildCount 1

點(diǎn)擊AddNum: 只打印了 re-render ChildNum 1

以上通過對(duì) Context 使用中的一些思考,我們簡單的實(shí)現(xiàn)了 useSelector,實(shí)現(xiàn)了 Context 組件的按需渲染

總結(jié)

在使用 Context API 的時(shí)候,要避免不必要的re-render,可以使用 useMemo 做細(xì)粒度更新,也可以使用 useSelector 實(shí)現(xiàn)按需渲染

以上就是React使用Context的一些優(yōu)化建議的詳細(xì)內(nèi)容,更多關(guān)于React Context的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • 詳解React中如何獲取真實(shí)的dom

    詳解React中如何獲取真實(shí)的dom

    這篇文章主要為大家詳細(xì)介紹了React中獲取真實(shí)的dom的相關(guān)方法,文中的示例代碼講解詳細(xì),具有一定的借鑒價(jià)值,感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下
    2025-02-02
  • react ant-design Select組件下拉框map不顯示的解決

    react ant-design Select組件下拉框map不顯示的解決

    這篇文章主要介紹了react ant-design Select組件下拉框map不顯示的解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2024-03-03
  • React Hooks使用方法全方位介紹

    React Hooks使用方法全方位介紹

    在react類組件(class)寫法中,有setState和生命周期對(duì)狀態(tài)進(jìn)行管理,但是在函數(shù)組件中不存在這些,故引入hooks(版本:>=16.8),使開發(fā)者在非class的情況下使用更多react特性
    2023-03-03
  • React的createElement和render手寫實(shí)現(xiàn)示例

    React的createElement和render手寫實(shí)現(xiàn)示例

    這篇文章主要為大家介紹了React的createElement和render手寫實(shí)現(xiàn)示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-08-08
  • react源碼層深入刨析babel解析jsx實(shí)現(xiàn)

    react源碼層深入刨析babel解析jsx實(shí)現(xiàn)

    同作為MVVM框架,React相比于Vue來講,上手更需要JavaScript功底深厚一些,本系列將閱讀React相關(guān)源碼,從jsx -> VDom -> RDOM等一些列的過程,將會(huì)在本系列中一一講解
    2022-10-10
  • React+echarts?(echarts-for-react)?實(shí)現(xiàn)中國地圖及省份切換功能

    React+echarts?(echarts-for-react)?實(shí)現(xiàn)中國地圖及省份切換功能

    這篇文章主要介紹了React+echarts?(echarts-for-react)?畫中國地圖及省份切換,有足夠的地圖數(shù)據(jù),可以點(diǎn)擊到街道,示例我只出到市級(jí),本文結(jié)合實(shí)例代碼給大家介紹的非常詳細(xì)需要的朋友可以參考下
    2022-11-11
  • 詳解在create-react-app使用less與antd按需加載

    詳解在create-react-app使用less與antd按需加載

    這篇文章主要介紹了詳解在create-react-app使用less與antd按需加載,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2018-12-12
  • 詳解react-router 4.0 下服務(wù)器如何配合BrowserRouter

    詳解react-router 4.0 下服務(wù)器如何配合BrowserRouter

    這篇文章主要介紹了詳解react-router 4.0 下服務(wù)器如何配合BrowserRouter,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2017-12-12
  • 如何使用React構(gòu)建一個(gè)高效的視頻上傳組件

    如何使用React構(gòu)建一個(gè)高效的視頻上傳組件

    本文介紹了如何使用React構(gòu)建一個(gè)高效的視頻上傳組件,包括基礎(chǔ)概念、常見問題與解決方案以及易錯(cuò)點(diǎn),通過實(shí)際代碼案例,展示了如何實(shí)現(xiàn)文件大小和格式驗(yàn)證、上傳進(jìn)度顯示等功能,并詳細(xì)解釋了跨域請(qǐng)求和并發(fā)上傳控制等技術(shù)細(xì)節(jié)
    2025-01-01
  • react-dnd實(shí)現(xiàn)任意拖動(dòng)與互換位置

    react-dnd實(shí)現(xiàn)任意拖動(dòng)與互換位置

    這篇文章主要為大家詳細(xì)介紹了react-dnd實(shí)現(xiàn)任意拖動(dòng)與互換位置,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-08-08

最新評(píng)論