[FE] React 初窥门径(七):hook 状态创建/更新原理

1. 回顾

上一篇 文章我们介绍了 React 函数组件的更新过程。
我们来总结以下 组件加载 和 更新 的全流程。

(1)组件载入时,会创建两棵 Fiber Tree

一棵为当前已写入 DOM 的 Fiber Tree(名为 current)。
commit 阶段 之前这个 Fiber Tree 只有一个根节点。

另一棵为当前正在渲染的 Fiber Tree,(名为 workInProgress)。
render 阶段 就是在创建它。

到了 commit 阶段,React 会将 workInProgress 的 Fiber Tree 实际写到 DOM 中,
然后将 current 指向这个 Fiber Tree。

这样就完成了组件的首次加载。

(2)事件触发组件更新时

首先是由 React 的事件系统监听到用户事件,然后触发用户绑定的事件处理函数。
在这个事件处理函数中,示例中我们用了 hook setState 来更新组件组件状态。
执行过程中,会将 performSyncWorkOnRoot 放到 syncQueue 中。
然后,用户事件就执行完了。

用户事件执行完之后,React 会紧接着执行 flushSyncCallbackQueue
获取到 syncQueue 中的 performSyncWorkOnRoot 进行执行。

performSyncWorkOnRoot 实际上就是组件的 rendercommit 方法。
(在组件的第一次更新时)它会创建一棵 workInProgress 的 Fiber Tree,然后在 commit 阶段 写到 DOM 中(之后,将 current 指向这棵 Fiber Tree)。
(如果是组件非首次更新,此时内存中已经有了两棵 Fiber Tree 了,此时 render 阶段,并不会重新创建一棵全新的 Fiber Tree,而是尽可能利用现有 Fiber Tree 的节点,这个逻辑在 createWorkInProgress 中控制)。

如此这般,就完成了组件的更新。

以上分析中,我们是从 Fiber Tree 的角度,从 rendercommit 的角度来看待组件的更新过程,
略过了组件的状态的计算过程。

在实际开发中,常见的场景是,

  • 有多个 hook(setState
  • 一次更新调用了多次 setState

React 内部是如何处理这个状态计算的呢?本文我们来仔细研究下这个问题。

2. 场景:多个 hook

2.1 示例项目的修改

参考 example-project/src/AppTwoState.js

我们修改了 App 组件如下,

const App = () => {
  debugger;
  const [state1, setState1] = useState(0);
  debugger;
  const [state2, setState2] = useState('a');
  debugger;

  const onDivClick = () => {
    debugger;
    setState1(1);
    debugger;
    setState2('b');
    debugger;
  };

  debugger;
  return 
{state1}-{state2}
; }

其中用到了两个 hook(都是 useState),这样会给 App 组件创建两个独立的状态 state1 state2

2.2 两个 hook 的更新流程

我们来跟踪一下两个 useStatesetState1 setState2 的执行过程。

完整的执行流程在这里:7.1 hook 原理:多个 hook,总共分为三个部分:

  • 组件首次加载时,调用 useState(第 1-50 行)

  • 用户点击 div 时,setState 调用 lastRenderedReducer 更新状态(第 51-96 行)

  • 事件响应完之前,React 调用 flushSyncCallbackQueue 更新状态(第 97-154 行)

2.3 多个 hook 是怎么存储的

我们看到组件载入的时候,useState 会调用 mountWorkInProgressHook

function mountWorkInProgressHook() {
  var hook = {
    memoizedState: null,
    baseState: null,
    baseQueue: null,
    queue: null,
    next: null
  };

  if (workInProgressHook === null) {
    // This is the first hook in the list
    currentlyRenderingFiber$1.memoizedState = workInProgressHook = hook;  // 第一个 useState
  } else {
    // Append to the end of the list
    workInProgressHook = workInProgressHook.next = hook;                  // 第二个 useState
  }

  return workInProgressHook;
}

每次调用 useState 会创建一个新的 hook,多个 hook 构成了一个链表结构(第二个 hooknext 指向 第一个 hook

(1)第一个 hook


currentlyRenderingFiber$1 节点(Fiber Node),
并且,Fiber Node 的 memorizedState 指向了 hook 链表的第一个 hook

currentlyRenderingFiber$1.memoizedState = workInProgressHook = hook;

(2)第二个 hook
设置 第一个 hooknext 属性指向 第二个 hook

通过 Fiber Node(currentlyRenderingFiber$1)观察一下 hook 链表的结构,

currentlyRenderingFiber$1.memorizedState -> hook1
hook1.next -> hook2
hook2.next -> null

2.4 dispatch(setState1 setState2)

虽然 hook 是通过链表结构来存储的,但实际调用 setState1 setState2 的时候,却并不是通过链表来取的。

这是是因为虽然 setState 只传入了一个参数 action

但实际 React 已通过 bind 传入了其他参数,另外两个参数是 fiberqueue

fiber 就是上文那个 currentlyRenderingFiber$1queue 就是 setState1 对应 hook 的 queue 属性值(hook 相关的 update quque,下文介绍)

所以调用 setState1 setState2 时不用在 hook 链表中进行查找,而是直接进入 dispatchAction 函数中。

3. 场景:多次 dispatch

上文介绍了多个 hook 的存储和调用原理,在实际项目中,还会有一个事件中多次调用了 dispatch(setState),
这些 dispatch 函数也许是同一个状态的 dispatch(多次调用 setState),也许是不同状态的(先后调用 setState1 setState2)。
原理其实是大同小异的,为了简单起见,本文只介绍后者,即,一个事件中,多次调用了同一个 hook 的 dispatch(setState)的执行流程。

3.1 示例项目的修改

示例项目的修改如下,example-project/src/AppAsyncState.js
(为了便于跟踪,setState 采用了回调方式进行编写)

const App = () => {
  debugger;
  const [state, setState] = useState(0);
  debugger;

  const onDivClick = () => {
    debugger;
    setState(s => {
      debugger;
      return s + 1;
    });
    debugger;
    setState(s => {
      debugger;
      return s + 2;
    });
    debugger;
  };

  debugger;
  return 
{state}
; }

3.2 多次调用 setState 的执行流程

完整的执行流程可参考 7.2 hook 原理:多次调用,包含以下两个部分,
(省略了组件首次加载的流程)

(1)用户点击 div 触发事件,事件中调用了两次 setState,第 1-65 行


我们看到 React 只执行了第一个状态更新函数(第一次 setStateaction 参数),

s => {
  debugger;
  return s + 1;
}

第二次 setStateaction 并未在这个阶段执行,而是将更新过程,放到了一个名为 update循环队列中。
参考 dispatchAction L16620

function dispatchAction(fiber, queue, action) {
  ...
  var update = {
    lane: lane,
    action: action,
    eagerReducer: null,
    eagerState: null,
    next: null
  };

  var pending = queue.pending;

  if (pending === null) {
    update.next = update;          // 第一次调用 setState 时,循环队列只有一个元素(自己指向自己)
  } else {
    update.next = pending.next;
    pending.next = update;         // <- 将 update 放到循环队列中(逻辑见下文解释)
  }

  queue.pending = update;
  var alternate = fiber.alternate;

  if (fiber === currentlyRenderingFiber$1 || alternate !== null && alternate === currentlyRenderingFiber$1) {
    ...
  } else {
    if (fiber.lanes === NoLanes && (alternate === null || alternate.lanes === NoLanes)) {
      ...
      if (lastRenderedReducer !== null) {
        ...
        try {
          var currentState = queue.lastRenderedState;
          var eagerState = lastRenderedReducer(currentState, action);

          update.eagerReducer = lastRenderedReducer;    // <- 用来标记这个 update 元素已经计算过了
          update.eagerState = eagerState;

          if (objectIs(eagerState, currentState)) {
            return;
          }
        } catch (error) {// Suppress the error. It will throw again in the render phase.
        } finally {
          ...
        }
      }
    }
    ...
  }
  ...
}

update 逻辑如下,

  • 每个 hook 维护了一个 update quque,hookpending 属性指向了这个 quque 的队尾(队尾的 next 为队首)
  • 每次调用 setState(= dispatchAction)都会创建一个 update 节点
  • 第一次调用 setState,update quque 只包含了一个元素(自己指向自己),然后设置 hook.pending 指向这个 update 元素
  • 第二次调用 setState,会在 update quque 队尾添加一个元素,再设置当前这个队尾元素指向队首,
hook.pending -> 当前的 update 元素
(当前的 update 元素).next -> 队首
原队尾.next -> 当前的 update 元素

以上这样设置的好处是,可以从队尾元素开始,循环获取 next 元素,将队列按顺序处理一遍。

值得一提的是,React 采用了给 update.eagerReducer 赋值为 lastRenderedReducer 的方式,来标记这个 update 元素已经处理过了,

update.eagerReducer = lastRenderedReducer;

这里要留意一下,下文会用到。

(2)事件完成之前,React 通过 flushSyncCallbackQueue,更新 Fiber Tree,并写入到 DOM 中,第 67-166 行


其中 syncQueue 中保存了 performSyncWorkOnRoot,React 用它在事件结束之前更新页面(见 前一篇 的分析)
update quque 是本文介绍的内容,React 在每次调用 setState 的时候,会创建一个循环队列,然后在 performSyncWorkOnRootrender 阶段 再执行计算。

代码逻辑在这里 updateReducer L15761

function updateReducer(reducer, initialArg, init) {
  var hook = updateWorkInProgressHook();
  var queue = hook.queue;
  ...
  if (baseQueue !== null) {
    ...
    do {                                        // <- 从队首开始处理 update quque
      ...
      if (!isSubsetOfLanes(renderLanes, updateLane)) {
        ...
      } else {
        ...
        if (update.eagerReducer === reducer) {  // 用来标记第一个 setState 已经计算过了
          newState = update.eagerState;
        } else {
          var action = update.action;
          newState = reducer(newState, action); // 后续未计算过的 setState,会按顺序执行计算
        }
      }

      update = update.next;
    } while (update !== null && update !== first);
    ...
  }
  ...
}

这里出现了对 update 元素 update.eagerReducer 的判定,来区分这个元素所表示的 setState 是否已经计算过了。
所以,除了第一个 setState 是 “同步”(setState 返回之前)执行的之外,
后续各个 setState 都是 “异步”(setState 返回后,由 React 通过 flushSyncCallbackQueuerender 阶段) 执行的。

4. fiber, hook, update

Fiber Tree,Fiber Node,hook,Update Queue 四者的关系如下,


  • Fiber Tree 有两棵
    一棵是已写入到的 DOM 的(称为 current),一棵是用于 render 阶段处理的(称为 workInProgress
    Fiber Tree 的根节点的 tagHostRoot
    两棵 Fiber Tree 的根节点通过 stateNode 指向 FiberRootNode,它通过 containerInfo 保存了 html 元素 div#root
    Fiber Tree 的节点有三个属性,return 指向父节点,child 指向子节点,alternate 指向同级的另一棵 Fiber Tree

  • 一个 React 组件可以使用多个 hook(创建多个独立的状态)
    hook 保存在了 Fiber Node (代表 元素的那个)的 memorizedState 属性中,多个 hook 以链表形式存储
    (同层级的 Fiber Node 共用一个 hook 对象)(可能会出现复制的情况)
    每一个 hook(比如 useState)返回一个新的 dispatch 方法,
    特定 dispatch 方法的每次调用,都会创建一个 update 元素,并添加到 update quque 中。
    hook.queue.pending 指向了 update queue 的队尾,队尾指向队首(循环队列)。

  • 组件通过 setState 进行状态更新时
    只有第一个 更新 会在 setState 返回值之前执行,不论是 setState(action) 中的 action 是数值还是函数
    后续所有(同一个或其他 hook)的 setState 调用,都会将更新放到 update quque 中,
    然后由 React 通过 flushSyncCallbackQueue 调用 performSyncWorkOnRootrender 阶段按顺序执行计算。


参考

React 初窥门径(六):React 组件的更新过程
github: thzt/react-tour
7.1 hook 原理:多个 hook
7.2 hook 原理:多次调用

你可能感兴趣的:([FE] React 初窥门径(七):hook 状态创建/更新原理)