一文读懂 React 组件渲染核心原理

开发 前端
相信大家对 React 都已经非常熟悉了,像 React,Vue 这样的现代前端框架已经是我们日常开发离不开的工具了,这篇文章主要是从源码的角度剖析 React 的核心渲染原理。

[[429160]]

引言

相信大家对 React 都已经非常熟悉了,像 React,Vue 这样的现代前端框架已经是我们日常开发离不开的工具了,这篇文章主要是从源码的角度剖析 React 的核心渲染原理。我们将从用户编写的组件代码开始,一步一步分析 React 是如何将它们变成真实 DOM ,这个过程主要可以分成两个阶段:render 阶段和 commit 阶段。文章的核心内容也正是对这两个阶段的分析。

一、前置知识

声明式渲染

『声明式渲染』,顾名思义,就是让使用者只需要「声明或描述」我需要渲染的东西是什么,然后就把具体的渲染工作交给机器去做,与之相对的是『命令式渲染』。

『命令式渲染』则是由用户去一步一步地命令机器下一步该怎么做。

举个简单的例子:

如果我们需要在网页上渲染一个有三个节点的列表,命令式的做法是手动操作 dom,首先创建一个容器节点,再利用循环每次先创建一个新节点,填充内容,然后将新节点新增到容器节点下,最后再将容器节点新增到 body 标签下:

  1. const list = [1,2,3]; 
  2. const container = document.createElement('div'); 
  3. for (let i = 0; i < list.length; i ++) { 
  4.     const newDom = document.createElement('div'); 
  5.     newDom.innerHTML = list[i]; 
  6.     container.appendChild(newDom); 
  7. document.body.appendChild(container); 

而声明式的做法应该是:

  1. const list = [1,2,3]; 
  2. const container = document.createElement('div'); 
  3.  
  4. const Demo = () => 
  5. (<div> 
  6.     {list.map((item) => <div>{item}</div>)} 
  7. </div>) 
  8.  
  9. ReactDom.render(<Demo />, container); 

可以看到在这个例子中,声明式写法以 HTML 语法直接告诉机器,我需要的视图应该是长这个样子,然后具体的 DOM 操作全部交由机器去完成。开发者只需要专注于业务逻辑的实现。

这便是声明式渲染。

声明式渲染是现代前端框架的比较普遍的设计思路。

JSX 和 ReactElement

相信大家最初学 React 的时候都有这样的疑问,为什么我们能够以类似 HTML 的语法编写组件,这个东西又是怎么转换成 JavaScript 语法的?答案就是 Babel。根据官网介绍,这种语法被称为 JSX,是一个 JavaScript 的语法扩展。能够被 Babel 编译成 React.createElement 方法。举个例子:

通过查阅源码我们可以看到 「React.createElement」 方法。

  1. export function createElement(type, config, children) { 
  2.   let propName; 
  3.  
  4.   // Reserved names are extracted 
  5.   const props = {}; 
  6.  
  7.   let key = null
  8.   let ref = null
  9.   let self = null
  10.   let source = null
  11.   ... 
  12.   return ReactElement( 
  13.     type, 
  14.     key
  15.     ref, 
  16.     self, 
  17.     source, 
  18.     ReactCurrentOwner.current
  19.     props, 
  20.   ); 
  21.  
  22. const ReactElement = function(type, key, ref, self, source, owner, props) { 
  23.   const element = { 
  24.     // This tag allows us to uniquely identify this as a React Element 
  25.     $typeof: REACT_ELEMENT_TYPE, 
  26.  
  27.     // Built-in properties that belong on the element 
  28.     type: type, 
  29.     keykey
  30.     ref: ref, 
  31.     props: props, 
  32.  
  33.     // Record the component responsible for creating this element. 
  34.     _owner: owner, 
  35.   }; 
  36.   ... 
  37.   return element; 

可以看到 React 是使用了 element 这种结构来代表一个节点,里面就只有简单的 6 个字段。我们可以看个实际的例子,下面 Count 组件对应的 element 数据结构:

  1. function Count({count, onCountClick}) { 
  2.   return <div onClick={() => { onCountClick()}}> 
  3.   count: {count
  4.   </div> 
  5.  
  6. <Count /> 

可以看到,element 结构只能反映出 jsx 节点的层级结构,而组件里的各种状态或者返回 jsx 等都是不会记录在 element 中。

目前我们知道,我们编写的 jsx 会首先被处理成 element 结构。

jsx -> element

那 React 又是如何处理 element 的,如刚刚说的,element 里包含的信息太少,只靠 element 显然是不足以映射到所有真实 DOM 的,因此我们还需要更精细的结构。

Fiber 树结构

Fiber 这个单词相信大家多多少少都有听过,它是在 React 16 被引入,关于 Fiber 如何实现任务调度在这篇文章不会涉及,但是 Fiber 的引入不仅仅带来了任务调度方面的能力,整个 React 实现架构也因此重构了一遍,而我们之前经常提到的虚拟 DOM 树在新的 React 架构下被称为 Fiber 树,上面提到的每个 element 都有一个所属的 Fiber。

首先我们先看看源码中 Fiber 的构造函数:

  1. function FiberNode( 
  2.   tag: WorkTag, 
  3.   pendingProps: mixed, 
  4.   keynull | string, 
  5.   mode: TypeOfMode, 
  6. ) { 
  7.   // Instance 
  8.   this.tag = tag;            // 标识节点类型,例如函数组件、类组件、普通标签等 
  9.   this.key = key
  10.   this.elementType = null;  // 标识具体 jsx 标签名 
  11.   this.type = null;        // 类似 elementType 
  12.   this.stateNode = null;  // 对应的真实 DOM 节点 
  13.  
  14.   // Fiber 
  15.   this.return = null;    // 父节点 
  16.   this.child = null;     // 第一个子节点 
  17.   this.sibling = null;   // 第一个兄弟节点 
  18.   this.index = 0; 
  19.  
  20.   this.ref = null
  21.  
  22.   this.pendingProps = pendingProps;  // 传入的 props 
  23.   this.memoizedProps = null;     
  24.   this.updateQueue = null;   // 状态更新相关 
  25.   this.memoizedState = null
  26.   this.dependencies = null
  27.  
  28.   this.mode = mode; 
  29.  
  30.   // Effects 
  31.   this.flags = NoFlags; 
  32.   this.subtreeFlags = NoFlags; 
  33.   this.deletions = null
  34.  
  35.   this.lanes = NoLanes; 
  36.   this.childLanes = NoLanes; 
  37.  
  38.   this.alternate = null
  39.   ... 

可以看到 Fiber 节点中的属性很多,其中不仅仅包含了 element 相关的实例信息,还包含了组成 Fiber 树所需的一些“指针”,组件内部的状态(memorizedState),用于操作真实 DOM 的副作用(effects)等等。

我们以上面的 Count 组件为例看一下它对应的 Fiber 结构:

这里我们先主要介绍一下与形成 Fiber 树相关的三个属性:child, sibling 和 return。他们分别指向 Fiber 的第一个子 Fiber,下一个兄弟 Fiber 和父 Fiber。

以下面的 jsx 代码为例:

  1. <App /> 
  2.       
  3. // App.jsx     
  4.     <div> 
  5.       <header> 
  6.         <img /> 
  7.         <p> 
  8.           text 
  9.         </p> 
  10.         <Count count={count} onCountClick={handleCLick} /> 
  11.       </header> 
  12.     </div> 
  13.      
  14. // Count.jsx 
  15. <div> 
  16. </div> 

最终形成的 Fiber 树结构为: 

总结一下,我们编写的 jsx 首先会形成 element ,然后在 render 过程中每个 element 都会生成对应的 Fiber,最终形成 Fiber 树。

jsx -> element -> Fiber

下面我们正式介绍一下 render 的过程,看看 Fiber 是如何生成并形成 Fiber 树的。

二、渲染(render)过程

核心流程

通常 React 运行时会有两个 Fiber 树,一个是根据当前最新组件状态构建出来的,另一个则是上一次构建出来的 Fiber 树,当然如果是首次渲染就没有上一次的 Fiber 树,这时就只有一个了。简单来说,render 过程就是 React 「对比旧 Fiber 树和新的 element」 然后「为新的 element 生成新 Fiber 树」的一个过程。

从源码中看,React 的整个核心流程开始于 「performSyncWorkOnRoot」 函数,在这个函数里会先后调用 「renderRootSync」 函数和 「commitRoot」 函数,它们两个就是分别就是我们上面提到的 render 和 commit 过程。来看 renderRootSync 函数,在 「renderRootSync」 函数里会先调用 「prepareFreshStack」 ,从函数名字我们不难猜出它主要就是为接下来的工作做前置准备,初始化一些变量例如 workInProgress(当前正在处理的 Fiber 节点) 等,接着会调用 「workLoopSync」 函数。(这里仅讨论传统模式,concurrent 模式留给 Fiber 任务调度分享),而在 「workLoopSync」 完成之后,「renderRootSync」 也基本上完成了,接下来就会调用 commitRoot 进入 commit 阶段。

因此整个 render 过程的重点在 「workLoopSync」 中,从 「workLoopSync」 简单的函数定义里我们可以看到,这里用了一个循环来不断调用 「performUnitOfWork」 方法,直到 workInProgress 为 null。

  1. function workLoopSync() { 
  2.   // Already timed out, so perform work without checking if we need to yield. 
  3.   while (workInProgress !== null) { 
  4.     performUnitOfWork(workInProgress); 
  5.   } 

而 「performUnitOfWork」 函数做的事情也很简单,简单来说就是为传进来的 workInProgress 生成下一个 Fiber 节点然后赋值给 workInProgress。通过不断的循环调用 「performUnitOfWork」,直到把所有的 Fiber 都生成出来并连接成 Fiber 树为止。

现在我们来看 「performUnitOfWork」 具体是如何生成 Fiber 节点的。

前面介绍 Fiber 结构的时候说过,Fiber 是 React 16 引入用于任务调度提升用户体验的,而在此之前,render 过程是递归实现的,显然递归是没有办法中断的,因此 React 需要使用循环来模拟递归过程。

「performUnitOfWork」 正是使用了 「beginWork」 和 「completeUnitOfWork」 来分别模拟这个“递”和“归”的过程。

render 过程是深度优先的遍历,「beginWork」 函数则会为遍历到的每个 Fiber 节点生成他的所有子 Fiber 并返回第一个子 Fiber ,这个子 Fiber 将赋值给 workInProgress,在下一轮循环继续处理,直到遍历到叶子节点,这时候就需要“归”了。

「completeUnitOfWork」 就会为叶子节点做一些处理,然后把叶子节点的兄弟节点赋值给 workInProgress 继续“递”操作,如果连兄弟节点也没有的话,就会往上处理父节点。

同样以上面的 Fiber 树例子来看,其中的 Fiber 节点处理顺序应该如下:

beginWork

在介绍概览的时候说过,React 通常会同时存在两个 Fiber 树,一个是当前视图对应的,一个则是根据最新状态正在构建中的。这两棵树的节点一一对应,我们用 current 来代表前者,我们不难发现,当首次渲染的时候,current 必然指向 null。实际上在代码中也确实都是通过这个来判断当前是首次渲染还是更新。

「beginWork」 的目的很简单:

  • 更新当前节点(workInProgress),获取新的 children。
  • 为新的 children 生成他们对应的 Fiber,并「最终返回第一个子节点(child)」。

在 「beginWork」 执行中,首先会判断当前是否是首次渲染。

  • 如果是首次渲染:
    • 则下来会根据当前正在构建的节点的组件类型做不同的处理,源码中这块逻辑使用了大量的 switch case。
  1. switch (workInProgress.tag) { 
  2.     case FunctionComponent: { 
  3.       ... 
  4.     } 
  5.     case ClassComponent: { 
  6.       ... 
  7.     } 
  8.     case HostRoot: { 
  9.       ... 
  10.     } 
  11.     case HostComponent: { 
  12.       ... 
  13.     } 
  14.     ... 
  15.   } 
  • 如果非首次渲染:
    • React 会使用一些优化手段,而符合优化的条件则是「当前节点对应组件的 props 和 context 没有发生变化」并且**当前节点的更新优先级不够,**如果这两个条件均满足的话可以直接复制 current 的子节点并返回。如果不满足则同首次渲染走一样的逻辑。
  1. if (current !== null) { 
  2.     // 这里处理一些依赖 
  3.     if ( 
  4.       enableLazyContextPropagation && 
  5.       !includesSomeLane(renderLanes, updateLanes) 
  6.     ) { 
  7.       const dependencies = current.dependencies; 
  8.       if (dependencies !== null && checkIfContextChanged(dependencies)) { 
  9.         updateLanes = mergeLanes(updateLanes, renderLanes); 
  10.       } 
  11.     } 
  12.  
  13.     const oldProps = current.memoizedProps; 
  14.     const newProps = workInProgress.pendingProps; 
  15.  
  16.     if ( 
  17.       oldProps !== newProps || 
  18.       hasLegacyContextChanged() || 
  19.       // Force a re-render if the implementation changed due to hot reload: 
  20.       (__DEV__ ? workInProgress.type !== current.type : false
  21.     ) { 
  22.       // 如果 props 或者 context 变了 
  23.       didReceiveUpdate = true
  24.     } else if (!includesSomeLane(renderLanes, updateLanes)) { 
  25.       didReceiveUpdate = false
  26.       // 走到这里则说明符合优化条件 
  27.       switch (workInProgress.tag) { 
  28.         case HostRoot: 
  29.           ... 
  30.           break; 
  31.         case HostComponent: 
  32.           ... 
  33.           break; 
  34.         case ClassComponent: { 
  35.           ... 
  36.           break; 
  37.         } 
  38.         case HostPortal: 
  39.           ... 
  40.           break; 
  41.         case ContextProvider: { 
  42.           ... 
  43.           break; 
  44.         } 
  45.         ... 
  46.          
  47.       } 
  48.       return bailoutOnAlreadyFinishedWork(current, workInProgress, renderLanes); 
  49.     } else { 
  50.       ... 
  51.       didReceiveUpdate = false
  52.     } 
  53.   } else { 
  54.     didReceiveUpdate = false
  55.   } 

更新优化策略应用

开发过程中我们常常希望利用 React 非首次渲染的优化策略来提升性能,如下代码,B 组件是个纯展示组件且内部没有依赖任何 Demo 组件的数据,因此有些同学可能会想当然认为当 Demo 重新渲染时这个 B 组件是符合 React 优化条件的。但结果是,每次 Demo 重新渲染都会导致 B 组件重新渲染。每次渲染时 B 组件的 props 看似没发生变化,但由于 Demo 重新执行后会生成全新的 B 组件(下面会介绍),所以新旧 B 组件的 props 肯定也是不同的。

  1. function App() { 
  2.     return <Demo /> 
  3.  
  4. function Demo() { 
  5.     const [v, setV] = useState(); 
  6.     return ( 
  7.         <div> 
  8.             <A value={v} /> 
  9.             <B /> 
  10.         </div>   
  11.     ); 

那有什么办法可以保持住 B 组件不变吗,答案是肯定的,我们可以把 B 组件放到 Demo 组件外层,这样一来,B 组件是在 App 组件中生成并作为 props 传入 Demo 的,因为不管 Demo 组件状态怎么变化都不会影响到 App 组件,因此 App 和 B 组件就只会在首次渲染时会执行一遍,也就是说 Demo 获取到的 props.children 的引用一直都是指向同一个对象,这样一来 B 组件的 props 也就不会变化了。

  1. function App() { 
  2.     return <Demo> 
  3.         <B /> 
  4.     </Demo> 
  5.  
  6. function Demo(props) { 
  7.     const [v, setV] = useState(); 
  8.     return ( 
  9.         <div> 
  10.             <A value={v} /> 
  11.            {props.children} 
  12.         </div>   
  13.     ); 

更新当前节点

通过上面的解析我们知道,当不走优化逻辑时 「beginWork」 使用大量的 switch...case 来分别处理不同类型的组件,下来我们以我们熟悉的 Function Component 为例。

「核心就是通过调用函数组件,得到组件的返回的 element。」

类似地,对于类组件,则是调用组件实例的 render 方法得到 element。

而对于我们普通的组件,例如<div></div> 则是直接取 props.children 即可。

  1. function updateFunctionComponent( 
  2.   current
  3.   workInProgress, 
  4.   Component, 
  5.   nextProps: any
  6.   renderLanes, 
  7. ) { 
  8.   let context; 
  9.   if (!disableLegacyContext) { 
  10.     const unmaskedContext = getUnmaskedContext(workInProgress, Component, true); 
  11.     context = getMaskedContext(workInProgress, unmaskedContext); 
  12.   } 
  13.   let nextChildren; 
  14.   prepareToReadContext(workInProgress, renderLanes); 
  15.   // 执行组件函数获取返回的 element 
  16.   nextChildren = renderWithHooks( 
  17.     current
  18.     workInProgress, 
  19.     Component, 
  20.     nextProps, 
  21.     context, 
  22.     renderLanes, 
  23.   ); 
  24.    
  25.   // React DevTools reads this flag. 
  26.   workInProgress.flags |= PerformedWork; 
  27.   reconcileChildren(current, workInProgress, nextChildren, renderLanes); 
  28.   return workInProgress.child; 

得到组件返回的 element(s) 之后,下一步就是为他们生成 Fiber,我们查看源码可以看到,不论是函数组件或是类组件或是普通组件,最后返回的 element(s) 都会作为参数传入到 「reconcileChildren」 中。

介绍 「reconcileChildren」 之前我们先用一张图总结一下 「beginWork」 的大致流程:

生成子节点

经过上一步得到 workInProgress 的 children 之后,接下来需要为这些 children element 生成 Fiber ,这就是 「reconcileChildFibers」 函数做的事情,这也是我们经常提到的 diff 的过程。

这个函数里主要分两种情况处理,如果是 newChild(即 children element)是 object 类型,则进入单节点 diff 过程(「reconcileSingleElement」),如果是数组类型,则进入多节点 diff 过程(「reconcileChildrenArray」)。

  1. function reconcileChildFibers( 
  2.     returnFiber: Fiber, 
  3.     currentFirstChild: Fiber | null
  4.     newChild: any
  5.     lanes: Lanes, 
  6.   ): Fiber | null { 
  7.     if (typeof newChild === 'object' && newChild !== null) { 
  8.       switch (newChild.$typeof) { 
  9.         case REACT_ELEMENT_TYPE: 
  10.           return placeSingleChild( 
  11.             reconcileSingleElement( 
  12.               returnFiber, 
  13.               currentFirstChild, 
  14.               newChild, 
  15.               lanes, 
  16.             ), 
  17.           ); 
  18.         ... 
  19.       } 
  20.       if (isArray(newChild)) { 
  21.         return reconcileChildrenArray( 
  22.           returnFiber, 
  23.           currentFirstChild, 
  24.           newChild, 
  25.           lanes, 
  26.         ); 
  27.       } 
  28.       throwOnInvalidObjectType(returnFiber, newChild); 
  29.     } 
  30.      
  31. }    

单节点diff

  1. function reconcileSingleElement( 
  2.     returnFiber: Fiber, 
  3.     currentFirstChild: Fiber | null
  4.     element: ReactElement, 
  5.     lanes: Lanes, 
  6.   ): Fiber { 
  7.     const key = element.key
  8.     let child = currentFirstChild; 
  9.     while (child !== null) { 
  10.      
  11.       // 首先比较 key 是否相同 
  12.       if (child.key === key) { 
  13.         const elementType = element.type; 
  14.         ... 
  15.            // 然后比较 elementType 是否相同 
  16.           if (child.elementType === elementType) { 
  17.             deleteRemainingChildren(returnFiber, child.sibling); 
  18.             const existing = useFiber(child, element.props); 
  19.             existing.ref = coerceRef(returnFiber, child, element); 
  20.             existing.return = returnFiber; 
  21.             return existing; 
  22.           } 
  23.          
  24.         // Didn't match. 
  25.         deleteRemainingChildren(returnFiber, child); 
  26.         break; 
  27.       } else { 
  28.         deleteChild(returnFiber, child); 
  29.       } 
  30.       // 遍历兄弟节点,看能不能找到 key 相同的节点 
  31.       child = child.sibling; 
  32.     } 
  33.  
  34.     if (element.type === REACT_FRAGMENT_TYPE) { 
  35.       const created = createFiberFromFragment( 
  36.         element.props.children, 
  37.         returnFiber.mode, 
  38.         lanes, 
  39.         element.key
  40.       ); 
  41.       created.return = returnFiber; 
  42.       return created; 
  43.     } else { 
  44.       const created = createFiberFromElement(element, returnFiber.mode, lanes); 
  45.       created.ref = coerceRef(returnFiber, currentFirstChild, element); 
  46.       created.return = returnFiber; 
  47.       return created; 
  48.     } 
  49.   } 
  50.    

本着尽可能复用旧节点的原则,在单节点 diff 在这里,我们会遍历旧节点,对每个遍历到的节点会做一下两个判断:

  • key 是否相同
  • key 相同的情况下,elementType 是否相同

延伸下来有三种情况:

  • 如果 key 不相同,则直接调用 「deleteChild」 将这个 child 标记为删除,但是我们不用灰心,可能只是我们还没有找到那个对的节点,所以要继续执行child = child.sibling;遍历兄弟节点,直到找到那个对的节点。
  • 如果 key 相同,elementType 相同,那就是最理想的情况,找到了可以复用的节点,直接调用 「deleteRemainingChildren」 把剩余的兄弟节点标记删除,然后直接复用 child 返回。
  • 如果 key 相同,但 elementType 不同,这是最悲情的情况,我们找到了那个节点,可惜的是这个节点的 elementType 已经变了,那我们也不需要再找了,把 child 及其所有兄弟节点标记删除,跳出循环。直接创建一个新的节点。

多节点diff

  1. function reconcileChildrenArray( 
  2.     returnFiber: Fiber, 
  3.     currentFirstChild: Fiber | null
  4.     newChildren: Array<*>, 
  5.     lanes: Lanes, 
  6. ) { 
  7.     let resultingFirstChild: Fiber | null = null
  8.     let previousNewFiber: Fiber | null = null
  9.  
  10.     let oldFiber = currentFirstChild; 
  11.     let lastPlacedIndex = 0; 
  12.     let newIdx = 0; 
  13.     let nextOldFiber = null
  14.     for (; oldFiber !== null && newIdx < newChildren.length; newIdx++) { 
  15.         const newFiber = updateSlot( 
  16.         returnFiber, 
  17.         oldFiber, 
  18.         newChildren[newIdx], 
  19.         lanes, 
  20.         ); 
  21.         if (newFiber === null) { 
  22.           break; 
  23.         } 
  24.           lastPlacedIndex = placeChild(newFiber, lastPlacedIndex, newIdx); 
  25.           if (previousNewFiber === null) { 
  26.             resultingFirstChild = newFiber; 
  27.           } else { 
  28.             previousNewFiber.sibling = newFiber; 
  29.           } 
  30.           previousNewFiber = newFiber; 
  31.           oldFiber = nextOldFiber; 
  32.     } 
  33.     if (newIdx === newChildren.length) { 
  34.         ... 
  35.     } 
  36.     if (oldFiber === null) { 
  37.         ... 
  38.     } 
  39.     for (; newIdx < newChildren.length; newIdx++) { 
  40.         ... 
  41.     } 
  42.     return resultingFirstChild; 
  43.  
  44.  
  45. function updateSlot( 
  46.     returnFiber: Fiber, 
  47.     oldFiber: Fiber | null
  48.     newChild: any
  49.     lanes: Lanes, 
  50.   ): Fiber | null { 
  51.     const key = oldFiber !== null ? oldFiber.key : null
  52.     ... 
  53.     if (newChild.key === key) { 
  54.       return updateElement(returnFiber, oldFiber, newChild, lanes); 
  55.     } else { 
  56.       return null
  57.     } 

从源码我们可以看到,在 「reconcileChildrenArray」 中,出现了两个循环。

第一轮循环中逻辑如下:

  • 同时遍历 oldFiber 链和 newChildren,判断 oldFiber 和 newChild 的 key 是否相同。
  • 如果 key 相同。
    • 判断双方 elementType 是否相同。
    • 如果相同则复用 oldFiber 返回。
    • 如果不同则新建 Fiber 返回。
  • 如果 key 不同则直接跳出循环。

可以看到第一轮循环只要碰到新旧的 key 不一样时就会跳出循环,换句话说,第一轮循环里做的事情都是基于 key 相同,主要就是「更新」的工作。

跳出循环后,要先执行两个判断:

  • newChildren 已经遍历完了:这种情况说明新的 children 全都已经处理完了,只要把 oldFiber 和他所有剩余的兄弟节点删除然后返回头部的 Fiber 即可。
  • 已经没有 oldFiber :这种情况说明 children 有新增的节点,给这些新增的节点逐一构建 Fiber 并链接上,然后返回头部的 Fiber 即可。

如果以上两种情况都不是,则进入第二轮循环。

在执行第二轮循环之前,先把剩下的旧节点和他们对应的 key 或者 index 做成映射,方便查找。

第二轮循环沿用了第一轮循环的 newIdx 变量,说明第二轮循环是在第一轮循环结束的地方开始再次遍历剩下的 newChildren。

  1.   const existingChildren = mapRemainingChildren(returnFiber, oldFiber); 
  2.   for (; newIdx < newChildren.length; newIdx++) { 
  3.     const newFiber = updateFromMap( 
  4.       existingChildren, 
  5.       returnFiber, 
  6.       newIdx, 
  7.       newChildren[newIdx], 
  8.       lanes, 
  9.     ); 
  10.     if (newFiber !== null) { 
  11.       if (shouldTrackSideEffects) { 
  12.         if (newFiber.alternate !== null) { 
  13.           existingChildren.delete
  14.             newFiber.key === null ? newIdx : newFiber.key
  15.           ); 
  16.         } 
  17.       } 
  18.       lastPlacedIndex = placeChild(newFiber, lastPlacedIndex, newIdx); 
  19.       if (previousNewFiber === null) { 
  20.         resultingFirstChild = newFiber; 
  21.       } else { 
  22.         previousNewFiber.sibling = newFiber; 
  23.       } 
  24.       previousNewFiber = newFiber; 
  25.     } 
  26.   } 
  27.    
  28.    
  29.   function placeChild( 
  30.   newFiber: Fiber, 
  31.   lastPlacedIndex: number, 
  32.   newIndex: number, 
  33. ): number { 
  34.   newFiber.index = newIndex; 
  35.   if (!shouldTrackSideEffects) { 
  36.     // Noop. 
  37.     return lastPlacedIndex; 
  38.   } 
  39.   const current = newFiber.alternate; 
  40.   if (current !== null) { 
  41.     const oldIndex = current.index
  42.     if (oldIndex < lastPlacedIndex) { 
  43.       // This is a move
  44.       newFiber.flags |= Placement; 
  45.       return lastPlacedIndex; 
  46.     } else { 
  47.       // This item can stay in place. 
  48.       return oldIndex; 
  49.     } 
  50.   } else { 
  51.     // This is an insertion. 
  52.     newFiber.flags |= Placement; 
  53.     return lastPlacedIndex; 
  54.   } 

第二轮循环主要调用了 「updateFromMap」 来处理节点,在这里需要用 newChild 的 key 去 existingChildren 中找对应的 Fiber。

  • 能找到 key 相同的,则说明这个节点只是位置变了,是可以复用的。
  • 找不到 key 相同的,则说明这个节点应该是新增的。

不管是复用还是新增,「updateFromMap」 都会返回一个 newFiber,然后我们需要为这个 newFiber 更新一下它的位置(index),但是仅仅更新这个 Fiber 的 index 还不够,因为这个 Fiber 有可能是复用的,如果是复用的就意味着它已经有对应的真实 DOM 节点了,我们还需要复用它的真实 DOM,因此需要对应更新这个 Fiber 的 flag,但是真的需要对每个 Fiber 都去设置 flag 吗,我们举个例子:

  1. // 旧 
  2. [<div key='a' />, <div key='b' />, <div key='c' />] 
  3.  
  4. // 新 
  5. [<div key='c' />, <div key='a' />, <div key='b' />] 

如果按照我们刚刚说的做法,这里的 a, b, c 都会被打上 flag,这样一来,在 commit 阶段,这三个 DOM 都会被移动,可是我们知道,这里显然只需要移动一个节点即可,退一万步说我们移动两个节点也比移动所有节点要来的聪明。

其实在这个问题上主要就是我们得区分一下到底哪个节点才是移动了的,这就需要一个参照点,我们要保证在参照点左边都是已经排好顺序了的。而这个参照点就是 lastPlacedIndex。有了它,我们在遍历 newChildren 的时候可能会出现下面两种情况:

  • 生成(或复用)的 Fiber 对应的老 index < lastPlacedIndex,这就说明这个 Fiber 的位置不对,因为 lastPlacedIndex 左边的应该全是已经遍历过的 newChild 生成的 Fiber。因此这个 Fiber 是需要被移动的,打上 flag。
  • 如果 Fiber 对应的老 index >= lastPlacedIndex,那就说明这个 Fiber 的相对位置是 ok 的,可以不用移动,但是我们需要更新一下参照点,把参照点更新成这个 Fiber 对应的老 index。

我们举一个例子:

  1. // 旧 
  2. [<div key='a' />, <div key='b' />, <div key='c' />, <div key='d' />] 
  3.  
  4. // 新 
  5. [<div key='c' />, <div key='a' />, <div key='b' />, <div key='d' />, <div key='e' />] 

lastPlacedIndex 初始值为 0,

首先处理第一个节点 c,给节点 c 的 index 赋值为最新值 0,c.index = 0。

可以看到 c 的 oldIndex 为 2,此时 oldIndex > lastPlacedIndex,无需对 c 做移动,将 lastPlacedIndex 赋值为 2。

此时 lastPlacedIndex = 2。

然后处理节点 a,a.index = 1。

a 的 oldIndex 为 0,此时 oldIndex < lastPlacedIndex,因此需要对 a 打上 Placement 标记,lastPlacedIndex 维持不变。

此时 lastPlacedIndex 仍然等于 2。

然后处理节点 b,b.index = 2。

b 的 oldIndex 为 1,此时 oldIndex < lastPlacedIndex,需要对 b 打上 Placement 标记,将 lastPlacedIndex 维持不变。

此时 lastPlacedIndex 仍然等于 2。

然后处理节点 d,b.index = 3。

d 的 oldIndex 为 3,此时 oldIndex > lastPlacedIndex,无需对 d 做移动,将 lastPlacedIndex 赋值为 3。

此时 lastPlacedIndex = 3。

然后处理节点 e,e.index = 4。

由于 e 是新建节点,所以 e 的 oldIndex 为 0,此时 oldIndex < lastPlacedIndex,因此需要对 e 打上 Placement 标记,lastPlacedIndex 维持不变。

因此最终需要变动位置的节点是 a b e。

这里可以看到其实最高效的改动是移动 c 和 e,但是 React 的 diff 逻辑选择了固定住 c,移动 a b,因此我们平时写代码的时候尽量避免把节点从后面提到前面的操作。

为 newChildren 里的所有 element 都生成了 Fiber 并连接好之后,返回第一个 child ,至此生成子节点的步骤就完成了。

completeUnitOfWork

在核心流程里我们说到,当 beginWork 处理到叶子节点,返回 null 的时候就会调用 「completeUnitOfWork」 函数。

「completeUnitOfWork」 主要做的事情有两件:

处理当前节点

  • “归”操作
  • 处理当前节点

「completeUnitOfWork」 里主要调用了 「completeWork」 来处理当前节点,而在 completeWork 中则是使用了 switch...case... 来处理不同类型的节点,这里我们主要以最常见的 HostComponent 为例。分成首次渲染和非首次渲染两种情况讨论。

mount

当是首次渲染时,这里要做的事情主要是:

  • 创建真实 DOM。
  • 如果有子节点的话将子节点的真实 DOM 插入到刚刚创建的 DOM 中。
  • 处理真实 DOM 的 props 等。
  1. const currentHostContext = getHostContext(); 
  2. // 为fiber创建对应DOM节点 
  3. const instance = createInstance( 
  4.     type, 
  5.     newProps, 
  6.     rootContainerInstance, 
  7.     currentHostContext, 
  8.     workInProgress, 
  9.   ); 
  10. // 将子孙DOM节点插入刚生成的DOM节点中 
  11. appendAllChildren(instance, workInProgress, falsefalse); 
  12. // DOM节点赋值给fiber.stateNode 
  13. workInProgress.stateNode = instance; 
  14.  
  15. // 处理props 
  16. if ( 
  17.   finalizeInitialChildren( 
  18.     instance, 
  19.     type, 
  20.     newProps, 
  21.     rootContainerInstance, 
  22.     currentHostContext, 
  23.   ) 
  24. ) { 
  25.   markUpdate(workInProgress); 

update

当 update 时,Fiber 节点已经存在对应 DOM 节点,所以不需要生成 DOM 节点。需要做的主要是处理DOM 节点的 props,这里主要就是一些真实 DOM 的 onClick、onChange等回调函数的注册,style 等,这些处理完之后的 props 也会记录到 workInProgress.updateQueue 中,并在 commit 阶段更新到 DOM 节点上。

  1. if (current !== null && workInProgress.stateNode != null) { 
  2.   // update的情况 
  3.   updateHostComponent( 
  4.     current
  5.     workInProgress, 
  6.     type, 
  7.     newProps, 
  8.     rootContainerInstance, 
  9.   ); 

“归”

刚刚说到,当 「beginWork」 返回值为 null 的时候会进入 「completeUnitOfWork」 中,可是我们知道 beginWork 是深度优先的更新,也就意味着进入 「completeUnitOfWork」 之后必然还需要回到 beginWork 中继续处理其他的节点。

  1. ... 
  2. const siblingFiber = completedWork.sibling; 
  3. if (siblingFiber !== null) { 
  4.   // If there is more work to do in this returnFiber, do that next
  5.   workInProgress = siblingFiber; 
  6.   return
  7. // Otherwise, return to the parent 
  8. completedWork = returnFiber; 
  9. // Update the next thing we're working on in case something throws. 
  10. workInProgress = completedWork; 

可以看到,当处理完当前节点之后,React 会判断当前节点是否具有兄弟节点,如果有的话则将兄弟节点设置为当前的 workInProgress 回到主流程继续 「beginWork。」

而如果没有兄弟节点的话,就意味着同父节点下的所有子节点都已经处理完毕,则接下来就会处理他们的父节点。

大致流程就是:「beginWork」 执行到当前节点没有 child 的时候,进入 「completeUnitOfWork」 处理当前节点,处理完后如果当前节点有兄弟节点则回到 「beginWork」 继续处理兄弟节点,如果没有兄弟节点则继续在 「completeUnitOfWork」 处理当前节点的父节点,直到“归”到根结点上。

三、挂载过程(commitRoot)

effect list

render阶段的一个主要工作是收集需要执行的 DOM 操作,然后交给 commit阶段 来处理,而这些 DOM 操作的具体类型都会保存在 Fiber 节点的 effectTag 属性上。

部分 DOM 操作的类型:

  1. // 插入 DOM 
  2. export const Placement = /*                */ 0b00000000000010; 
  3. // 更新 DOM 
  4. export const Update = /*                   */ 0b00000000000100; 
  5. // 插入并更新 DOM 
  6. export const PlacementAndUpdate = /*       */ 0b00000000000110; 
  7. // 删除 DOM 节点 
  8. export const Deletion = /*                 */ 0b00000000001000; 

使用二进制来表示可以方便地用位运算给 effectTag 带上多个副作用(effect),也可以方便地判断是否存在某个副作用。例如有个 fiber 节点 effectTag 是 PlacementAndUpdate (0b00000000000110),可以通过按位与运算来判断是否存在Placement:

  1. const effectTag = PlacementAndUpdate; 
  2. console.log(effectTag & Placement !== 0); // => true 

在 commit阶段 可以像 render阶段 那样遍历所有 fiber 节点找出其中的 effectTag,但这样效率比较低,所以在 render阶段 的 completeUnitOfWork 中会把具有 effectTag 的 fiber 节点连接起来,形成 effectList 链表。例如我们有这样的代码:

  1. function App() { 
  2.   const [count, setCount] = useState(0); 
  3.   return ( 
  4.     <div onClick={() => setCount(count + 1)}> 
  5.       <p>{count}</p> 
  6.       <span>{count}</span> 
  7.     </div> 
  8.   ); 
  9. }; 

我们在 performSyncWorkOnRoot 方法的末尾的 finishedWork 打个断点,然后点击 div 触发一次 setCount 更新:

当 react 执行到断点这个地方时,我们在控制台打印一下 finishedWork:

所以此时对应的 effectList是:

这里 span 和 p 节点有 effectTag 是因为 {count},div 节点有 effectTag 是因为重新生成的 onClick 函数。

commit 阶段

render阶段结束后,会在performSyncWorkOnRoot()或finishConcurrentRender()中把 fiberRootNode 传给 commitRoot 方法,开启 commit阶段。

以performSyncWorkOnRoot为例:

  1. function performSyncWorkOnRoot(root) { 
  2.     // render阶段的入口函数 
  3.     renderRootSync(root, lanes); 
  4.     // ... 
  5.     // commitRoot函数调用 
  6.     const finishedWork = root.current.alternate; 
  7.     root.finishedWork = finishedWork; 
  8.     root.finishedLanes = lanes; 
  9.     commitRoot(root); 
  10.     // ...省略代码 

commit阶段的一个主要工作就是遍历 effectList 并执行对应的 DOM 操作。commit阶段又分为三个子阶段:

  • before mutation阶段
  • mutation阶段
  • layout阶段

下面来看看 commit阶段 具体发生了什么。

进入 commitRootImpl 方法时,会先判断 rootWithPendingPassiveEffects 是否为 null,如果不为 null 就会执行 flushPassiveEffects。

  1. function commitRootImpl(root, renderPriorityLevel) { 
  2.   do { 
  3.     flushPassiveEffects(); 
  4.   } while (rootWithPendingPassiveEffects !== null); 
  5.  
  6.   // ... 

rootWithPendingPassiveEffects 中的PassiveEffect 是什么意思呢?我们知道如果一个 fiber 节点的 dom 节点需要被插入到页面中,那 fiber.effectTag 就会带上 Placement effect,类似的,如果一个 FunctionComponent 有 useEffect 需要被执行,那它就会带上 Passive effect。

所以这里的意思是进入 commitRoot 时先判断当前是否还有未执行的 useEffect,如果有,就执行它,也就是说在开启新一轮的 commit 阶段时会先等待上一轮的 useEffect 执行完。这其实在官方文档里也有一些说明:

接着会重置 render阶段使用到的一些全局变量:

  1. function commitRootImpl(root, renderPriorityLevel) { 
  2.   do { 
  3.     flushPassiveEffects(); 
  4.   } while (rootWithPendingPassiveEffects !== null); 
  5.  
  6.   // ... 
  7.  
  8.   if (root === workInProgressRoot) { 
  9.     workInProgressRoot = null
  10.     workInProgress = null
  11.     workInProgressRootRenderLanes = NoLanes; 
  12.   } 
  13.    
  14.   // ... 

处理 effect list:

  1. function commitRootImpl(root, renderPriorityLevel) { 
  2.   // ... 
  3.  
  4.   let firstEffect; 
  5.   if (finishedWork.effectTag > PerformedWork) { 
  6.     if (finishedWork.lastEffect !== null) { 
  7.       finishedWork.lastEffect.nextEffect = finishedWork; 
  8.       firstEffect = finishedWork.firstEffect; 
  9.     } else { 
  10.       firstEffect = finishedWork; 
  11.     } 
  12.   } else { 
  13.     firstEffect = finishedWork.firstEffect; 
  14.   } 
  15.    
  16.   // ... 

上面说过 render阶段已经把带有 effectTag 的 fiber 节点连接形成一条链表了,这里再次处理 effect list 是因为这条链表目前只有子节点,并没有挂载根节点。如果根节点也存在 effectTag,那么就需要把根节点拼接到链表的末尾,形成一条完整的 effect list:

同时上面的代码也会取出 firstEffect,也就是第一个需要被处理的 fiber 节点。接着判断如果存在 firstEffect,会将 firstEffect 赋值给 nextEffect,开始三个子阶段的工作。

  1. function commitRootImpl(root, renderPriorityLevel) { 
  2.   // ... 
  3.  
  4.   if (firstEffect !== null) { 
  5.     nextEffect = firstEffect; 
  6.      
  7.     // beforeMutation 阶段: 
  8.     // 执行 commitBeforeMutationEffects 
  9.      
  10.     // mutation 阶段: 
  11.     // 执行 commitMutationEffects 
  12.      
  13.     // layout 阶段: 
  14.     // 执行 commitLayoutEffects 
  15.   } 
  16.    
  17.   // ... 

执行 getSnapshotBeforeUpdate

在commitBeforeMutationEffects这个方法中会遍历带有 effectTag 的 fiber 节点,如果判断有 Snapshot effectTag 就会调用 ClassComponent 的 getSnapshotBeforeUpdate 生命周期方法:

  1. function commitBeforeMutationEffects() { 
  2.   while (nextEffect !== null) { 
  3.     // ... 
  4.      
  5.     const effectTag = nextEffect.effectTag; 
  6.     if ((effectTag & Snapshot) !== NoEffect) { 
  7.       // ... 
  8.       commitBeforeMutationEffectOnFiber(current, nextEffect); 
  9.     } 
  10.  
  11.     // ... 
  12.     nextEffect = nextEffect.nextEffect; 
  13.   } 

调度 useEffect

再看下 commitBeforeMutationEffects 的剩余部分:

  1. function commitBeforeMutationEffects() { 
  2.   while (nextEffect !== null) { 
  3.     // ... 
  4.      
  5.     if ((effectTag & Passive) !== NoEffect) { 
  6.       if (!rootDoesHavePassiveEffects) { 
  7.         rootDoesHavePassiveEffects = true
  8.         scheduleCallback(NormalSchedulerPriority, () => { 
  9.           flushPassiveEffects(); 
  10.           return null
  11.         }); 
  12.       } 
  13.     } 
  14.     nextEffect = nextEffect.nextEffect; 
  15.   } 

上面提到 flushPassiveEffects 用于执行 useEffect 的回调函数,而这里并不会立即执行它,而是把它放在 scheduleCallback 的回调当中,scheduleCallback 方法会以一个优先级异步执行它的回调函数。

所以这段代码的意思是,如果存在 Passive effect,则把 rootDoesHavePassiveEffects 置为 true,并且调度 flushPassiveEffects,而整个 commit阶段 是「同步执行」的,所以 useEffect 的回调函数其实会在 commit阶段「完成后」再异步执行。

这也跟官方文档说的对应上了:

总结

在 beforeMutation阶段 会:

  • 执行getSnapshotBeforeUpdate生命周期方法
  • 「调度」useEffect

mutation阶段

mutation阶段会执行 commitMutationEffects 方法:

  1. function commitMutationEffects(root: FiberRoot, renderPriorityLevel) { 
  2.   while (nextEffect !== null) { 
  3.     const effectTag = nextEffect.effectTag; 
  4.      
  5.     // 如果有 ContentReset,会重置文本节点 
  6.     if (effectTag & ContentReset) { 
  7.       commitResetTextContent(nextEffect); 
  8.     } 
  9.      
  10.     // 如果有 Ref,会执行 ref 相关的更新 
  11.     if (effectTag & Ref) { 
  12.       // ... 
  13.     } 
  14.  
  15.     const primaryEffectTag = 
  16.       effectTag & (Placement | Update | Deletion | Hydrating); 
  17.     switch (primaryEffectTag) { 
  18.       // 如果需要插入节点,会执行 commitPlacement 
  19.       case Placement: { 
  20.         commitPlacement(nextEffect); 
  21.         nextEffect.effectTag &= ~Placement; 
  22.         break; 
  23.       } 
  24.       // 如果需要更新节点,会执行 commitWork 
  25.       case Update: { 
  26.         const current = nextEffect.alternate; 
  27.         commitWork(current, nextEffect); 
  28.         break; 
  29.       } 
  30.       // 如果需要删除节点,会执行 commitDeletion 
  31.       case Deletion: { 
  32.         commitDeletion(root, nextEffect, renderPriorityLevel); 
  33.         break; 
  34.       } 
  35.       // ... 
  36.     } 
  37.      
  38.     // 取出下一个 fiber 节点,进入下一次循环 
  39.     nextEffect = nextEffect.nextEffect; 
  40.   } 

在这个方法中会遍历带有 effectTag 的 fiber 节点,

  • 如果有 ContentReset,会重置文本节点
  • 如果有 Ref,会执行 ref 相关的操作
  • 增删改
    • 如果需要插入节点,会执行 commitPlacement
    • 如果需要更新节点,会执行 commitWork
    • 如果需要删除节点,会执行 commitDeletion
    • ...

commitPlacement

在 commitPlacement 方法中,会先找到距离最近的 host 类型父节点和距离最近的 host 类型兄弟节点,然后根据 host 父 fiber 节点的类型取出对应的 DOM 节点,接着根据是否 container 来执行 insertOrAppendPlacementNodeIntoContainer 或 insertOrAppendPlacementNode。

  1. function commitPlacement(finishedWork: Fiber): void { 
  2.   // 找到 host 父 fiber 节点 
  3.   const parentFiber = getHostParentFiber(finishedWork); 
  4.  
  5.   let parent; 
  6.   let isContainer; 
  7.   const parentStateNode = parentFiber.stateNode; 
  8.    
  9.   // 根据 host 父 fiber 节点的类型,取出对应的 DOM 节点 
  10.   switch (parentFiber.tag) { 
  11.     case HostComponent: 
  12.       parent = parentStateNode; 
  13.       isContainer = false
  14.       break; 
  15.     case HostRoot: 
  16.       parent = parentStateNode.containerInfo; 
  17.       isContainer = true
  18.       break; 
  19.     case HostPortal: 
  20.       parent = parentStateNode.containerInfo; 
  21.       isContainer = true
  22.       break; 
  23.     // ... 
  24.   } 
  25.   // ... 
  26.   // 获取 host 兄弟节点 
  27.   const before = getHostSibling(finishedWork); 
  28.    
  29.   // 根据是否 container 来决定执行哪个方法 
  30.   if (isContainer) { 
  31.     insertOrAppendPlacementNodeIntoContainer(finishedWork, before, parent); 
  32.   } else { 
  33.     insertOrAppendPlacementNode(finishedWork, before, parent); 
  34.   } 

为什么需要先找到 host 父 fiber 节点和 host 兄弟 fiber 节点?

我们知道在DOM中插入一个节点有两种方式:

  • parentNode.appendChild(newNode)
  • parentNode.insertBefore(newNode, referenceNode)

无论哪种方式都需要找到它的父 DOM 节点,而如果需要 insertBefore 则还需要找到它的兄弟 DOM 节点。

另外,为什么需要找 host 类型的父节点和兄弟节点呢?这是因为最近的父 「fiber」 节点不一定就是最近的父 「DOM」 节点,同理,最近的兄弟 「fiber」 节点不一样是最近的兄弟 「DOM」 节点。例如,我们的代码长这样:

  1. function Item() { 
  2.   return <p></p> 
  3.  
  4. function App() { 
  5.   return ( 
  6.     <div> 
  7.       <Item /> 
  8.       <span></span> 
  9.     </div> 
  10.   ); 

它对应的 fiber 树和 dom 树分别是:

可以看到, fiber 树和 don 树并不是一一对应的:

  • p 的 fiber 节点的父节点是 Item,而 p 的 dom 节点的父节点是 div
  • p 的 fiber 节点没有兄弟节点,而 p 的 dom 节点有个兄弟节点是 span

上面的insertOrAppendPlacementNodeIntoContainer 和 insertOrAppendPlacementNode做的事情差不多,都会判断是否有 before,如果有,则执行 insertBefore,没有则执行 appendChild:

  1. function insertOrAppendPlacementNode( 
  2.   node: Fiber, 
  3.   before: ?Instance, 
  4.   parent: Instance, 
  5. ): void { 
  6.   // ... 
  7.   if (before) { 
  8.     insertBefore(parent, stateNode, before); 
  9.   } else { 
  10.     appendChild(parent, stateNode); 
  11.   } 
  12.   // ... 

appendChild 和 insertBefore 都来自于 ReactFiberHostConfig。

这里的 ReactFiberHostConfig 在源码里其实只是一个空壳,最终需要被特定环境的 renderer 来填充,例如在我们平常使用 ReactDOM 时,ReactFiberHostConfig 会被 ReactDOM 的 ReactDOMHostConfig 来填充:

  1. export function appendChild( 
  2.   parentInstance: Instance, 
  3.   child: Instance | TextInstance, 
  4. ): void { 
  5.   parentInstance.appendChild(child); 
  6.  
  7. export function insertBefore( 
  8.   parentInstance: Instance, 
  9.   child: Instance | TextInstance, 
  10.   beforeChild: Instance | TextInstance | SuspenseInstance, 
  11. ): void { 
  12.   parentInstance.insertBefore(child, beforeChild); 

可以看到,其实最终就是执行 dom 节点的 appendChild 或 insertBefore 方法。

React 仓库其他 renderer 的 hostConfig:

BTW,umijs/hox(https://github.com/umijs/hox) 是一个基于 hooks 的全局状态管理工具,它跟其他基于 hooks 的状态管理工具很大的一个不同点是, hox 不需要我们手动挂载 就能直接使用 model:

  1. // counterModel.js 
  2. import { useState } from "react"
  3. import { createModel } from "hox"
  4. function useCounter() { 
  5.   const [count, setCount] = useState(0); 
  6.   const decrement = () => setCount(count - 1); 
  7.   const increment = () => setCount(count + 1); 
  8.   return { 
  9.     count
  10.     decrement, 
  11.     increment 
  12.   }; 
  13. export default createModel(useCounter); 
  14.  
  15. // index.jsx 
  16. import useCounterModel from "./counterModel"
  17. function App(props) { 
  18.   const counter = useCounterModel(); 
  19.   return ( 
  20.     <div> 
  21.       <p>{counter.count}</p> 
  22.       <button onClick={counter.increment}>Increment</button> 
  23.     </div> 
  24.   ); 

其实它是在调用 createModel 时把 hooks 的执行挂在了它自定义的一个 renderer 里,对应的 hostConfig 都是空函数,因为它并不需要执行真正的渲染,只是用来执行 hooks 而已:

  1. import ReactReconciler from "react-reconciler"
  2. import { ReactElement } from "react"
  3.  
  4. const hostConfig = { 
  5.   now: Date.now, 
  6.   getRootHostContext: () => ({}), 
  7.   prepareForCommit: () => {}, 
  8.   resetAfterCommit: () => {}, 
  9.   getChildHostContext: () => ({}), 
  10.   shouldSetTextContent: () => true
  11.   createInstance: () => {}, 
  12.   createTextInstance: () => {}, 
  13.   appendInitialChild: () => {}, 
  14.   appendChild: () => {}, 
  15.   finalizeInitialChildren: () => {}, 
  16.   supportsMutation: true
  17.   appendChildToContainer: () => {}, 
  18.   prepareUpdate: () => true
  19.   commitUpdate: () => {}, 
  20.   commitTextUpdate: () => {}, 
  21.   removeChild: () => {} 
  22. }; 
  23.  
  24. const reconciler = ReactReconciler(hostConfig as any); 
  25.  
  26. export function render(reactElement: ReactElement) { 
  27.   const container = reconciler.createContainer(nullfalsefalse); 
  28.   return reconciler.updateContainer(reactElement, container, nullnull); 

commitWork

commitWork 用于更新节点。在这个方法中会根据 fiber 节点的类型进行不同的操作:

  1. function commitWork(current: Fiber | null, finishedWork: Fiber): void { 
  2.   switch (finishedWork.tag) { 
  3.     case FunctionComponent: 
  4.     case ForwardRef: 
  5.     case MemoComponent: 
  6.     case SimpleMemoComponent: 
  7.     case Block: { 
  8.       // ... 
  9.       return
  10.     } 
  11.     case ClassComponent: { 
  12.       return
  13.     } 
  14.     case HostComponent: { 
  15.       // ... 
  16.       return
  17.     } 
  18.     case HostText: { 
  19.       // ... 
  20.       return
  21.     } 
  22.     // case... 
  23.   } 

对于 FunctionComponent,更新时会执行 commitHookEffectListUnmount(``HookLayout | HookHasEffect, finishedWork):

  1. function commitHookEffectListUnmount(tag: number, finishedWork: Fiber) { 
  2.   const updateQueue = finishedWork.updateQueue; 
  3.   const lastEffect = updateQueue !== null ? updateQueue.lastEffect : null
  4.   if (lastEffect !== null) { 
  5.     const firstEffect = lastEffect.next
  6.     let effect = firstEffect; 
  7.     do { 
  8.       if ((effect.tag & tag) === tag) { 
  9.         const destroy = effect.destroy; 
  10.         effect.destroy = undefined; 
  11.         if (destroy !== undefined) { 
  12.           destroy(); 
  13.         } 
  14.       } 
  15.       effect = effect.next
  16.     } while (effect !== firstEffect); 
  17.   } 

里面会遍历 effect list,判断 effect.tag 是否存在参数传入的 tag 类型,在这个场景里被传入 tag 参数是 HookLayout,也就是 useLayoutEffect 对应的 effect tag。所以这里的意思是遍历 effect list,如果存在 useLayoutEffect 的 effect tag,则执行它的销毁函数(即 useLayoutEffect 的回调函数的返回值)。

如果 fiber 节点的类型是 HostComponent,也就是 dom 节点对应的 fiber 节点,更新时会执行 commitUpdate 方法:

  1. function commitWork(current: Fiber | null, finishedWork: Fiber): void { 
  2.   switch (finishedWork.tag) { 
  3.     // ... 
  4.     case HostComponent: { 
  5.       const instance: Instance = finishedWork.stateNode; 
  6.       if (instance != null) { 
  7.         // ... 
  8.         const updatePayload: null | UpdatePayload = (finishedWork.updateQueue: any); 
  9.         // ... 
  10.         if (updatePayload !== null) { 
  11.           commitUpdate( 
  12.             instance, 
  13.             updatePayload, 
  14.             type, 
  15.             oldProps, 
  16.             newProps, 
  17.             finishedWork, 
  18.           ); 
  19.         } 
  20.         if (enableDeprecatedFlareAPI) { 
  21.           const prevListeners = oldProps.DEPRECATED_flareListeners; 
  22.           const nextListeners = newProps.DEPRECATED_flareListeners; 
  23.           if (prevListeners !== nextListeners) { 
  24.             updateDeprecatedEventListeners(nextListeners, finishedWork, null); 
  25.           } 
  26.         } 
  27.       } 
  28.       return
  29.     } 
  30.     // ... 
  31.   } 

commitUpdate 方法接收的参数中有个 updatePayload,它来自于 fiber.updateQueue 属性,对于类型为 HostComponent 的 fiber 节点来说,它的 updateQueue 属性是一个数组,表示这个 dom 节点的属性变更,例如一个 dom 节点在某次更新中它的 a 属性需要从 react 更新为 vue,b 属性需要从 byte 更新为 dance,那这个 dom 节点的 fiber 节点的 updateQueue 就长这样 ['a', 'vue', 'b', 'dance'],也就是说第 i 项是属性 key,第 i + 1 项是属性 value。

  1. export function commitUpdate( 
  2.   domElement: Instance, 
  3.   updatePayload: Array<mixed>, 
  4.   type: string, 
  5.   oldProps: Props, 
  6.   newProps: Props, 
  7.   internalInstanceHandle: Object, 
  8. ): void { 
  9.   // ... 
  10.   updateProperties(domElement, updatePayload, type, oldProps, newProps); 

它最终会调用 updateDOMProperties 来更新 dom 属性:

  1. function updateDOMProperties( 
  2.   domElement: Element, 
  3.   updatePayload: Array<any>, 
  4.   wasCustomComponentTag: boolean, 
  5.   isCustomComponentTag: boolean, 
  6. ): void { 
  7.   for (let i = 0; i < updatePayload.length; i += 2) { 
  8.     const propKey = updatePayload[i]; 
  9.     const propValue = updatePayload[i + 1]; 
  10.     if (propKey === STYLE) { 
  11.       setValueForStyles(domElement, propValue); 
  12.     } else if (propKey === DANGEROUSLY_SET_INNER_HTML) { 
  13.       setInnerHTML(domElement, propValue); 
  14.     } else if (propKey === CHILDREN) { 
  15.       setTextContent(domElement, propValue); 
  16.     } else { 
  17.       setValueForProperty(domElement, propKey, propValue, isCustomComponentTag); 
  18.     } 
  19.   } 

这里的 i 和 i + 1 就对应着上面提到的 updateQueue 的数据结构,第 i 项是属性 key,第 i + 1 项是属性 value。

commitDeletion

commitDeletion 用于执行删除操作:

  1. function commitDeletion( 
  2.   finishedRoot: FiberRoot, 
  3.   current: Fiber, 
  4.   renderPriorityLevel: ReactPriorityLevel, 
  5. ): void { 
  6.   // ... 
  7.   unmountHostComponents(finishedRoot, current, renderPriorityLevel); 
  8.   // ... 

在 unmountHostComponents 中核心是遍历节点调用 commitUnmount 方法。在这个方法中会根据 fiber 节点的类型做不同的处理。

对于 FunctionComponent,会注册它的 useEffect 销毁函数,其实就是把这个 effect 推进 pendingPassiveHookEffectsUnmount 这个数组中,便于「后续」取出来执行销毁函数。

  1. function commitUnmount( 
  2.   finishedRoot: FiberRoot, 
  3.   current: Fiber, 
  4.   renderPriorityLevel: ReactPriorityLevel, 
  5. ): void { 
  6.   switch (current.tag) { 
  7.     case FunctionComponent: { 
  8.       // ... 
  9.       enqueuePendingPassiveHookEffectUnmount(current, effect); 
  10.       // ... 
  11.       return
  12.     } 
  13.   } 

对于 ClassComponent,会执行它的 componentWillUnmount 方法:

  1. function commitUnmount( 
  2.   finishedRoot: FiberRoot, 
  3.   current: Fiber, 
  4.   renderPriorityLevel: ReactPriorityLevel, 
  5. ): void { 
  6.   switch (current.tag) { 
  7.     case ClassComponent: { 
  8.       const instance = current.stateNode; 
  9.       if (typeof instance.componentWillUnmount === 'function') { 
  10.         safelyCallComponentWillUnmount(current, instance); 
  11.       } 
  12.       return
  13.     } 
  14.   } 

layout阶段

layout阶段 会执行 commitLayoutEffects 方法,里面核心是执行commitLifeCycles方法。在这个方法中会根据 fiber 节点的类型执行不同的处理。

  1. function commitLifeCycles( 
  2.   finishedRoot: FiberRoot, 
  3.   current: Fiber | null
  4.   finishedWork: Fiber, 
  5.   committedLanes: Lanes, 
  6. ): void { 
  7.   switch (finishedWork.tag) { 
  8.     case FunctionComponent: { 
  9.       commitHookEffectListMount(HookLayout | HookHasEffect, finishedWork); 
  10.       schedulePassiveEffects(finishedWork); 
  11.       return
  12.     } 
  13.   } 

对于 FunctionComponent,会把 HookLayout 这个 tag 类型传给 commitHookEffectListMount 方法,也就是说这里会执行 useLayoutEffect 的回调函数。

接着会执行 schedulePassiveEffects 方法:

  1. function schedulePassiveEffects(finishedWork: Fiber) { 
  2.   const updateQueue: FunctionComponentUpdateQueue | null = (finishedWork.updateQueue: any); 
  3.   const lastEffect = updateQueue !== null ? updateQueue.lastEffect : null
  4.   if (lastEffect !== null) { 
  5.     const firstEffect = lastEffect.next
  6.     let effect = firstEffect; 
  7.     do { 
  8.       const {next, tag} = effect; 
  9.       if ( 
  10.         (tag & HookPassive) !== NoHookEffect && 
  11.         (tag & HookHasEffect) !== NoHookEffect 
  12.       ) { 
  13.         enqueuePendingPassiveHookEffectUnmount(finishedWork, effect); 
  14.         enqueuePendingPassiveHookEffectMount(finishedWork, effect); 
  15.       } 
  16.       effect = next
  17.     } while (effect !== firstEffect); 
  18.   } 

在这里会分别注册 useEffect 销毁函数和回调函数,其实也就是把 effect 分别推进 pendingPassiveHookEffectsUnmount 和 pendingPassiveHookEffectsMount 这两个数组中,用于「后续」取出来执行。

对于 ClassComponent,如果 current 为空,也就是这个节点是首次 render,则会执行它的 componentDidMount 生命周期方法,否则会执行 componentDidUpdate 方法:

  1. function commitLifeCycles( 
  2.   finishedRoot: FiberRoot, 
  3.   current: Fiber | null
  4.   finishedWork: Fiber, 
  5.   committedLanes: Lanes, 
  6. ): void { 
  7.   switch (finishedWork.tag) { 
  8.     case ClassComponent: { 
  9.       const instance = finishedWork.stateNode; 
  10.       if (finishedWork.effectTag & Update) { 
  11.         if (current === null) { 
  12.           instance.componentDidMount(); 
  13.         } else { 
  14.           instance.componentDidUpdate( 
  15.             prevProps, 
  16.             prevState, 
  17.             instance.__reactInternalSnapshotBeforeUpdate, 
  18.           ); 
  19.         } 
  20.       } 
  21.       return
  22.     } 
  23.   } 

commit 阶段总结

  • 等待执行完上一轮渲染的 useEffect
  • 重置一些全局变量(如:workInProgressRoot)
  • 更新副作用列表 effect list。根节点的副作用列表是不包括自身的,如果根节点有副作用, 则需要把根节点添加到副作用列表的末尾
  • 渲染
    • 执行 componentDidMount、componentDidUpdate
    • 执行 useLayoutEffect 的回调函数
    • 注册 useEffect 的回调函数和销毁函数,等 commit 阶段结束后再异步执行
    • 执行 DOM 增删改
    • 执行 useLayoutEffect 的销毁函数
    • 执行 getSnapshotBeforeUpdate
    • 调度 useEffect
    • beforeMutation
    • mutation
    • 切换当前 fiber 树(root.current = finishedWork),使得 fiberRoot 的 current 指向的是当前页面展示的 fiber 树。
    • layout
责任编辑:武晓燕 来源: Tecvan
相关推荐

2021-12-16 14:45:09

https架构服务端

2023-01-09 08:14:08

GoHttpServer

2022-05-12 10:53:42

keepalivevrrp协议

2021-04-27 11:28:21

React.t事件元素

2024-01-09 08:24:47

JMM核心线程

2021-08-04 16:06:45

DataOps智领云

2023-12-22 19:59:15

2017-12-21 14:48:43

数据安全数据泄漏检测技术

2021-10-20 07:18:51

Linux延时队列

2021-02-26 05:24:35

Java垃圾回收

2021-04-30 19:53:53

HugePages大内存页物理

2018-09-28 14:06:25

前端缓存后端

2022-09-22 09:00:46

CSS单位

2022-11-06 21:14:02

数据驱动架构数据

2024-04-10 10:34:34

Cache系统GPU

2021-03-10 21:43:11

5G核心网

2022-04-10 10:57:06

eBPFJIT即时编译

2020-05-14 14:52:05

HDFS数据集架构

2022-10-20 08:01:23

2023-05-20 17:58:31

低代码软件
点赞
收藏

51CTO技术栈公众号