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

詳解react中useCallback內(nèi)部是如何實現(xiàn)的

 更新時間:2023年07月04日 09:11:32   作者:AKclown  
前幾天有人在問在useCallback函數(shù)如果第二個參數(shù)為空數(shù)組, 為什么拿不到最新的state值,那么這一章就來分析一下useCallback內(nèi)部是如何實現(xiàn)的,感興趣的小伙伴跟著小編一起來學(xué)習(xí)吧

示例demo與debug

新建了一個react項目,將APP.tsx改寫成如下代碼

import  { useCallback, useState } from 'react';
function App() {
  const [num, updateNum] = useState(0);
  const TestCallback  = useCallback(() =>{
    console.log('num: ', num);
  },[]);
  return (
    <div className="App">
        <p onClick={() => {
          updateNum(num => num + 1);
          updateNum(num => num + 1);
          updateNum(num => num + 1);
        }}>{num}</p>
        <p onClick={TestCallback}>打印</p>
    </div>
  );
}
export default App;

在瀏覽器的source設(shè)置斷點,熟悉一遍useCallback的調(diào)用流程。(由于.gif過大,這里就不上git了,自行調(diào)試)

源碼解析

useCallback的整體流程框架

在react中mount階段和update階段進入到同一個useCallback方法里。但resolveDispatcher找到的dispatch對象mountupdate會不同,最終導(dǎo)致在mount階段調(diào)用mountCallback而update階段調(diào)用的是updateCallback。
下面為調(diào)用useCallback方法觸發(fā)的行為

function useCallback(callback, deps) {
  var dispatcher = resolveDispatcher();
  return dispatcher.useCallback(callback, deps);
}

下面來看看resolveDispatcher是如何獲取到dispatch的

function resolveDispatcher() {
  const dispatcher = ReactCurrentDispatcher.current;
  ...
  return ((dispatcher: any): Dispatcher);
}

ReactCurrentDispatcher.current會在renderWithHooks方法中進行所處階段判斷并且賦值。如果current === null || current.memoizedState === null為true表示在mount階段反正為update階段

function renderWithHooks<Props, SecondArg>(...) {
     ...
     ReactCurrentDispatcher.current =
      current === null || current.memoizedState === null
        ? HooksDispatcherOnMount
        : HooksDispatcherOnUpdate;
}
// mount階段調(diào)用的dispatch
const HooksDispatcherOnMount: Dispatcher = {
  ...
  useCallback: mountCallback,
};
// update階段調(diào)用的dispatch
const HooksDispatcherOnUpdate: Dispatcher = {
  ...
  useCallback: updateCallback,
};

從上面的代碼分析可以知道在mounted階段調(diào)用的是mountCallbackupdate階段調(diào)用updateCallback

Hook

一個函數(shù)式組件鏈路: fiber(FunctionComponent) => Hook(保存數(shù)據(jù)狀態(tài)) => Queue(更新的隊列結(jié)構(gòu)) => update(更新的數(shù)據(jù))在后續(xù)需要使用到Hook這個結(jié)構(gòu),那么先來看一下Hook是數(shù)據(jù)結(jié)構(gòu)是怎么樣的,以及屬性的作用是什么?

  • memoizedState 存放的是Hook對應(yīng)的state
  • next鏈接到下一個Hook,從而形成一個無環(huán)單向鏈表
  • queue存儲同一個hook更新的多個update對象,數(shù)據(jù)結(jié)構(gòu)為環(huán)狀單向鏈表
// 組件對應(yīng)的fiber對象
const fiber = {
  // 保存該FunctionComponent對應(yīng)的Hooks鏈表
  memoizedState: hook,
  ...
};
const hook: Hook = {
    // 1.  memoizedState 存放的是Hook對應(yīng)的state
    memoizedState: null,
    // 2. next鏈接到下一個Hook,從而形成一個`無環(huán)單向鏈表`
    queue: null,
    // 3. queue存儲同一個hook更新的多個update對象,數(shù)據(jù)結(jié)構(gòu)為`環(huán)狀單向鏈表`  
    next: null,
    ...
};

fiber與Hooks的關(guān)系(懶得畫圖了,引用了Understanding the Closure Trap of React Hooks)

mount階段

分析mountCallback的實現(xiàn)

  • 通過mountWorkInProgressHook獲取到對應(yīng)的Hook對象
  • 判斷條件deps是否為undefined
  • 回調(diào)函數(shù)判斷條件存入到hook.memoizedState
  • 返回傳入的回調(diào)函數(shù)
function mountCallback<T>(callback: T, deps: Array<mixed> | void | null): T {
  const hook = mountWorkInProgressHook();
  const nextDeps = deps === undefined ? null : deps;
  hook.memoizedState = [callback, nextDeps];
  return callback;
}

mountWorkInProgressHook的實現(xiàn),創(chuàng)建初始化Hook對象,并且將該Hook對象保存在workInProgressHook鏈路中. workInProgressHook表示正在執(zhí)行的hook

function mountWorkInProgressHook(): Hook {
  const hook: Hook = {
    memoizedState: null,
    baseState: null,
    baseQueue: null,
    queue: null,
    next: null,
  };
  if (workInProgressHook === null) {
    // This is the first hook in the list
    currentlyRenderingFiber.memoizedState = workInProgressHook = hook;
  } else {
    // Append to the end of the list
    workInProgressHook = workInProgressHook.next = hook;
  }
  return workInProgressHook;
}

在組件render時,每當(dāng)遇到下一個Hook,通過移動workInProgressHook的指針來獲取到對應(yīng)的HookPS: 只要每次組件renderuseState的調(diào)用順序及數(shù)量保持一致,那么始終可以通過workInProgressHook找到當(dāng)前useState對應(yīng)的hook對象

// fiber.memoizedState標識第一個Hook
workInProgressHook = fiber.memoizedState;
// 在組件`render`時,遇到下一個hook時
workInProgressHook = workInProgressHook.next;
....

update階段

分析updateCallback的實現(xiàn)

  • 通過updateWorkInProgressHook獲取到當(dāng)前的Hook對象
  • hook.memoizedState獲取到上一次緩存的state。假設(shè)這是第一次update那么其值就是mount階段保存的[callback, nextDeps]數(shù)據(jù)
  • 如果依賴條件不為空,使用areHookInputsEqual判斷依賴項是否更改。只會遍歷數(shù)組第一層數(shù)據(jù)比較不會做深層比較。如果依賴項沒變化,返回原本緩存的callback
function updateCallback<T>(callback: T, deps: Array<mixed> | void | null): T {
  const hook = updateWorkInProgressHook();
  const nextDeps = deps === undefined ? null : deps;
  const prevState = hook.memoizedState;
  if (nextDeps !== null) {
    const prevDeps: Array<mixed> | null = prevState[1];
    if (areHookInputsEqual(nextDeps, prevDeps)) {
      return prevState[0];
    }
  }
  hook.memoizedState = [callback, nextDeps];
  return callback;
}

依賴比較areHookInputsEqual的方法實現(xiàn)

function areHookInputsEqual(
  nextDeps: Array<mixed>,
  prevDeps: Array<mixed> | null,
): boolean {
  ...
  // $FlowFixMe[incompatible-use] found when upgrading Flow
  for (let i = 0; i < prevDeps.length && i < nextDeps.length; i++) {
    // $FlowFixMe[incompatible-use] found when upgrading Flow
    if (is(nextDeps[i], prevDeps[i])) {
      continue;
    }
    return false;
  }
  return true;
}

總結(jié)

在React中會使用閉包機制來處理上文的callback回調(diào)函數(shù)。當(dāng)包含useCallback組件被渲染時,React 會為該特定渲染周期創(chuàng)建一個閉包。閉包是一個封裝的作用域,其中包含渲染時位于作用域內(nèi)的變量、函數(shù)和其他引用。
因此deps我們傳入的是空數(shù)組,其回調(diào)函數(shù)callback一直引用的狀態(tài)始終是初始狀態(tài),無法獲取最新狀態(tài)。緩存的回調(diào)函數(shù)可以訪問最初調(diào)用時范圍內(nèi)的狀態(tài)和道具

插件推薦

閱讀源碼可以通過使用Bookmarks快速標記代碼位置,實現(xiàn)快速條件

到此這篇關(guān)于詳解react中useCallback內(nèi)部是如何實現(xiàn)的的文章就介紹到這了,更多相關(guān)react useCallback內(nèi)部實現(xiàn)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • React性能debug場景解決記錄

    React性能debug場景解決記錄

    這篇文章主要為大家介紹了React性能debug場景解決記錄,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-05-05
  • React中異步數(shù)據(jù)更新不及時問題及解決

    React中異步數(shù)據(jù)更新不及時問題及解決

    這篇文章主要介紹了React中異步數(shù)據(jù)更新不及時問題及解決方案,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2024-03-03
  • react中使用Modal.confirm數(shù)據(jù)不更新的問題完美解決方案

    react中使用Modal.confirm數(shù)據(jù)不更新的問題完美解決方案

    這篇文章主要介紹了react中使用Modal.confirm數(shù)據(jù)不更新的問題解決方案,本文通過實例代碼給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2023-09-09
  • React函數(shù)組件和類組件的區(qū)別及說明

    React函數(shù)組件和類組件的區(qū)別及說明

    這篇文章主要介紹了React函數(shù)組件和類組件的區(qū)別及說明,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-08-08
  • 使用webpack搭建react開發(fā)環(huán)境的方法

    使用webpack搭建react開發(fā)環(huán)境的方法

    本篇文章主要介紹了使用webpack搭建react開發(fā)環(huán)境的方法,在這篇文章中我們開始利用我們之前所學(xué)搭建一個簡易的React開發(fā)環(huán)境,用以鞏固我們之前學(xué)習(xí)的Webpack知識。一起跟隨小編過來看看吧
    2018-05-05
  • React Native預(yù)設(shè)占位placeholder的使用

    React Native預(yù)設(shè)占位placeholder的使用

    本篇文章主要介紹了React Native預(yù)設(shè)占位placeholder的使用,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-09-09
  • React項目中使用Redux的?react-redux

    React項目中使用Redux的?react-redux

    這篇文章主要介紹了React項目中使用Redux的?react-redux,文章圍繞主題展開詳細的內(nèi)容介紹,具有一定的參考價值,需要的小伙伴可以參考一下
    2022-09-09
  • React中的useEffect(副作用)介紹

    React中的useEffect(副作用)介紹

    這篇文章主要介紹了React中的useEffect(副作用),具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2024-01-01
  • React自定義Hook的實現(xiàn)

    React自定義Hook的實現(xiàn)

    自定義Hook是一種自定義函數(shù),它封裝了React Hook的邏輯,并通過命名約定來使其可重用,本文主要介紹了React自定義Hook的實現(xiàn),感興趣的可以了解一下
    2023-11-11
  • 使用 React Router Dom 實現(xiàn)路由導(dǎo)航的詳細過程

    使用 React Router Dom 實現(xiàn)路由導(dǎo)航的詳細過程

    React Router Dom 是 React 應(yīng)用程序中用于處理路由的常用庫,它提供了一系列組件和 API 來管理應(yīng)用程序的路由,這篇文章主要介紹了使用 React Router Dom 實現(xiàn)路由導(dǎo)航,需要的朋友可以參考下
    2024-03-03

最新評論