React essence note - 14 -ReactDOM.render 是如何串联渲染链路的(上)

ReactDOM.render 是如何串联渲染链路的(上)

将以首次渲染为切入点, 拆解Fiber架构下ReactDOM.render()所触发的渲染链路,结合源码理解整个链路中所涉及的初始化,rendercommit等过程.

ReactDOM.render 调用栈的逻辑分层

运行react基础示例,打开performance工具,初次渲染得到下面的调用栈图

放大该图,定位src/index.js这个文件路径,我们就可以找到 ReactDOM.render 方法对应的调用栈,如下图所示":

图中 scheduleUpdateOnFiber 方法的作用是调度更新, 在由 ReactDOM.render 发起的首屏渲染这个场景下, 它触发的就是 performSyncWorkOnRoot. performSyncWorkOnRoot 开启的正是我们反复强调的 render 阶段; 而 commitRoot 方法开启的则是真实 DOM 的渲染过程(commit 阶段). 因此以scheduleUpdateOnFibercommitRoot 两个方法为界, 我们可以大致把 ReactDOM.render 的调用栈划分为三个阶段:

  • 初始化阶段
  • render 阶段
  • commit 阶段

拆解 ReactDOM.render 调用栈——初始化阶段

上图完成了Fiber树中基本实体的创建

什么是基本实体? 基本实体有哪些?

  • 首先是legacyRenderSubtreeIntoContainer方法,调用方式
    return legacyRenderSubtreeIntoContainer(null, element, container, false, callback)

    // 方法源码
    function legacyRenderSubtreeIntoContainer(parentComponent, children, container, forceHydrate, callback) {
        // container 对应的是我们传入的真实 DOM 对象
        var root = container._reactRootContainer;
        // 初始化 fiberRoot 对象
        var fiberRoot;
        // DOM 对象本身不存在 _reactRootContainer 属性, 因此 root 为空
        if (!root) {
            // 若 root 为空, 则初始化 _reactRootContainer, 并将其值赋值给 root
            root = container._reactRootContainer = legacyCreateRootFromDOMContainer(container, forceHydrate);
            // legacyCreateRootFromDOMContainer 创建出的对象会有一个 _internalRoot 属性, 将其赋值给 fiberRoot
            fiberRoot = root._internalRoot;
            // 这里处理的是 ReactDOM.render 入参中的回调函数, 你了解即可
            if (typeof callback === 'function') {
            var originalCallback = callback;
            callback = function () {
                var instance = getPublicRootInstance(fiberRoot);
                originalCallback.call(instance);
            };
            } // Initial mount should not be batched.
            // 进入 unbatchedUpdates 方法
            unbatchedUpdates(function () {
            updateContainer(children, fiberRoot, parentComponent, callback);
            });
        } else {
            // else 逻辑处理的是非首次渲染的情况(即更新), 其逻辑除了跳过了初始化工作, 与楼上基本一致
            fiberRoot = root._internalRoot;
            if (typeof callback === 'function') {
            var _originalCallback = callback;
            callback = function () {
                var instance = getPublicRootInstance(fiberRoot);
                _originalCallback.call(instance);
            };
            } // Update
            updateContainer(children, fiberRoot, parentComponent, callback);
        }
        return getPublicRootInstance(fiberRoot);
    }
  • 首次渲染过程中 legacyRenderSubtreeIntoContainer 方法的主要逻辑链路

在这个流程中, 你需要关注到 fiberRoot 这个对象. fiberRoot 到底是什么呢? 这里我将运行时的 rootfiberRoot 为你截取出来, 其中 root 对象的结构如下图所示

可以看出, root 对象(container._reactRootContainer)上有一个 _internalRoot 属性, 这个 _internalRoot 也就是 fiberRoot. fiberRoot 的本质是一个 FiberRootNode 对象, 其中包含一个 current 属性, 该属性同样需要划重点. 这里我为你高亮出 current 属性的部分内容

或许你会对 current 对象包含的海量属性感到陌生和头大, 但这并不妨碍你 Get 到current 对象是一个 FiberNode 实例这一点, 而 FiberNode, 正是 Fiber 节点对应的对象类型, current 对象是一个 Fiber 节点, 不仅如此, 它还是当前 Fiber 树的头部节点.

考虑到 current 属性对应的 FiberNode 节点, 在调用栈中实际是由 createHostRootFiber 方法创建的, React 源码中也有多处以 rootFiber 代指 current 对象, 因此下文中我们将以 rootFiber 指代 current 对象. 其对应关系

其中, fiberRoot 的关联对象是真实 DOM 的容器节点; 而 rootFiber 则作为虚拟 DOM 的根节点存在. 这两个节点, 将是后续整棵 Fiber 树构建的起点.

接下来, fiberRoot 将和 ReactDOM.render 方法的其他入参一起, 被传入 updateContainer 方法, 从而形成一个回调. 这个回调, 正是接下来要调用的 unbatchedUpdates 方法的入参. 我们一起看看 unbatchedUpdates 做了什么, 下面代码是对 unbatchedUpdates 主体逻辑的提取

function unbatchedUpdates(fn, a) {
  // 这里是对上下文的处理, 不必纠结
  var prevExecutionContext = executionContext;
  executionContext &= ~BatchedContext;
  executionContext |= LegacyUnbatchedContext;
  try {
    // 重点在这里, 直接调用了传入的回调函数 fn, 对应当前链路中的 updateContainer 方法
    return fn(a);
  } finally {
    // finally 逻辑里是对回调队列的处理, 此处不用太关注
    executionContext = prevExecutionContext;
    if (executionContext === NoContext) {
      // Flush the immediate callbacks that were scheduled during this batch
      resetRenderTimer();
      flushSyncCallbackQueue();
    }
  }
}

unbatchedUpdates 函数体里, 当下你只需要 Get 到一个信息:它直接调用了传入的回调 fn. 而在当前链路中, fn 是什么呢? fn 是一个针对 updateContainer 的调用

unbatchedUpdates(function () {
  updateContainer(children, fiberRoot, parentComponent, callback);
});

updateContainer 里面的逻辑

function updateContainer(element, container, parentComponent, callback) {
  ......
  // 这是一个 event 相关的入参, 此处不必关注
  var eventTime = requestEventTime();
  ......
  // 这是一个比较关键的入参, lane 表示优先级
  var lane = requestUpdateLane(current$1);
  // 结合 lane(优先级)信息, 创建 update 对象, 一个 update 对象意味着一个更新
  var update = createUpdate(eventTime, lane); 
  // update 的 payload 对应的是一个 React 元素
  update.payload = {
    element: element
  };
  // 处理 callback, 这个 callback 其实就是我们调用 ReactDOM.render 时传入的 callback
  callback = callback === undefined ? null : callback;
  if (callback !== null) {
    {
      if (typeof callback !== 'function') {
        error('render(...): Expected the last optional `callback` argument to be a ' + 'function. Instead received: %s.', callback);
      }
    }
    update.callback = callback;
  }
  // 将 update 入队
  enqueueUpdate(current$1, update);
  // 调度 fiberRoot 
  scheduleUpdateOnFiber(current$1, lane, eventTime);
  // 返回当前节点(fiberRoot)的优先级
  return lane;
}

updateContainer 的逻辑相对来说丰富了点, 但大部分逻辑也是在干杂活, 它做的最关键的事情可以总结为三件:

  1. 请求当前 Fiber 节点的 lane(优先级)
  2. 结合 lane(优先级), 创建当前 Fiber 节点的 update 对象, 并将其入队
  3. 调度当前节点(rootFiber)

函数体中的 scheduleWork 其实就是 scheduleUpdateOnFiber, scheduleUpdateOnFiber 函数的任务是调度当前节点的更新. 在这个函数中, 会处理一系列与优先级、打断操作相关的逻辑. 但是在 ReactDOM.render 发起的首次渲染链路中, 这些意义都不大, 因为这个渲染过程其实是同步的.

performSyncWorkOnRoot 直译过来就是 执行根节点的同步任务, 这里的 同步二字需要注意, 它明示了接下来即将开启的是一个同步的过程. 这也正是为什么在整个渲染链路中, 调度(Schedule)动作没有存在感的原因.

前面我们曾经提到过, performSyncWorkOnRoot 是 render 阶段的起点, render 阶段的任务就是完成 Fiber 树的构建, 它是整个渲染链路中最核心的一环.在异步渲染的模式下, render 阶段应该是一个可打断的异步过程(下一讲我们就将针对 render 过程作详细的逻辑拆解).

同步的 ReactDOM.render,异步的 ReactDOM.createRoot

都说 Fiber 架构带来的异步渲染是 React 16 的亮点,为什么分析到现在,竟然发现 ReactDOM.render 触发的首次渲染是个同步过程呢? ====> 同步的 ReactDOM.render,异步的 ReactDOM.createRoot
在 React 16,包括近期发布的 React 17 小版本中,React 都有以下 3 种启动方式

  1. legacy 模式: ReactDOM.render(<App />, rootNode). 这是当前 React App 使用的方式,当前没有计划删除本模式,但是这个模式可能不支持这些新功能.
  2. blocking 模式: ReactDOM.createBlockingRoot(rootNode).render(<App />). 目前正在实验中,作为迁移到 concurrent 模式的第一个步骤.
  3. concurrent 模式: ReactDOM.createRoot(rootNode).render(<App />). 目前在实验中, 未来稳定之后, 打算作为 React 的默认开发模式, 这个模式开启了所有的新功能.

在这 3 种模式中, 我们常用的 ReactDOM.render 对应的是 legacy 模式, 它实际触发的仍然是同步的渲染链路.blocking 模式可以理解为 legacyconcurrent 之间的一个过渡形态, 之所以会有这个模式, 是因为 React 官方希望能够提供渐进的迁移策略, 帮助我们更加顺滑地过渡到 Concurrent 模式.blocking 在实际应用中是比较低频的一个模式, 了解即可.

按照官方的说法, “长远来看, 模式的数量会收敛, 不用考虑不同的模式, 但就目前而言, 模式是一项重要的迁移策略, 让每个人都能决定自己什么时候迁移, 并按照自己的速度进行迁移”.由此可以看出, Concurrent 模式确实是 React 的终极目标, 也是其创作团队使用 Fiber 架构重写核心算法的动机所在.

Fiber 架构一定是异步渲染吗?

React 16 如果没有开启 Concurrent 模式, 那它还能叫 Fiber 架构吗?: 从动机上来看, Fiber 架构的设计确实主要是为了 Concurrent 而存在.但经过了本讲紧贴源码的讲解, 相信你也能够看出, 在 React 16, 包括已发布的 React 17 版本中, 不管是否是 Concurrent, 整个数据结构层面的设计、包括贯穿整个渲染链路的处理逻辑, 已经完全用 Fiber 重构了一遍.站在这个角度来看, Fiber 架构在 React 中并不能够和异步渲染画严格的等号, 它是一种同时兼容了同步渲染与异步渲染的设计.

总结

我们以 ReactDOM.render 所触发的首次渲染为切入点, 试图串联 React Fiber 架构下完整的工作链路, 本讲为整个源码链路分析的前半部分.

虽然当前的进度条只推到了初始化这个位置, 但在这部分的分析过程中, 相信你已经对 Fiber 树的初始形态、Fiber 根节点的创建过程建立了感性的认知, 同时把握住了 ReactDOM.render 同步渲染的过程特征, 理解了 React 当下共存的3种渲染方式. 在此基础上, 我们再去理解 render 过程, 就会轻松得多.

整个初始化的工作过程都是在为后续的 render 阶段做准备. 现在, 我们的 Fiber Tree 还处在只有根节点的起始状态. 接下来, 我们就要进入到最最关键的 render 阶段里去

posted @ 2020-11-23 15:52  荣光无限  阅读(146)  评论(0)    收藏  举报