当我们调用 setState 之后发生了什么?react经历了怎样的过程将新的 state 渲染到页面上?
一次react更新,核心就是对虚拟dom进行diff,找出最少的需要变化的dom节点,然后对其进行相应的dom操作,用户即可在页面上看到更新。但 react 作为广泛使用的框架,需要考虑更多的因素,考虑多个更新的优先级,考虑主线程占用时长,考虑diff算法复杂度,考虑性能。。等等,本文就来探讨一下react在其内部是如何处理数据更新的。
react在内部使用fiber这种数据结构来作为虚拟dom【react16+】,它与dom tree一一对应,形成fiber tree,一次react更新,本质是fiber tree结构的更新变化。而fiber tree结构的更新,用更专业的术语来讲,其实就是fiber tree的协调(Reconcile)。Reconcile中文意思是调和、使一致,协调fiber tree,就是调整fiber tree的结构,使其和更新后的jsx模版结构、dom tree保持一致。
react从16起,将更新机制分为三个模块,也可以说是三个步骤,分别是Schedule【调度】、Reconcile【协调】、render【渲染】
Schedule
为什么需要Schedule?
首先我们要知道react在进行协调时,提供了两种模式:Legacy mode 同步阻塞模式和 Concurrent mode 并行模式。
不同上下文中的更新会触发不同的模式,如果是在 event、setTimeout、network request 的 callback 中触发更新,react 会采用 Legacy 模式。如果更新与 Suspense、useTransition、OffScreen 相关,那么 react 会采用 Concurrent 模式。
Legacy mode
Legacy mode在协调时会启动 workLoopSync。workLoopSync 开始工作以后,要等到所有 fiber node 都处理完毕以后,才会结束工作,也就是 fiber tree 的协调过程不可中断。
Legacy mode存在的问题:如果 fiber tree 的结构很复杂,那么协调 fiber tree 可能会占用大量的时间,导致主线程会一直被 js 引擎占用,渲染引擎无法在规定时间(浏览器刷新频率 - 16.7ms)内完成工作,使得页面出现卡顿(掉帧),影响用户体验。
Concurrent mode
鉴于Legacy mode存在的问题,react团队在react 16中提出了 Concurrent mode的概念,并在react 18中开放使用。react16、17一直为此做准备。
Concurrent 模式最大的意义在于,使用Concurrent 模式以后的react的应用可以做到:
- 协调可以中断、恢复;不会长时间阻塞浏览器渲染
- 高优先级更新可以中断低优先级更新,优先渲染
那么,怎么做到这两点呢?
事实上,Schedule就是用来完成这个任务的,调度任务的优先级,使高优先级任务优先进入Reconcile,并且提供中断和恢复机制。
时间切片
react采用时间切片的方式来实现协调的中断和恢复,Concurrent mode在协调时会启动 workLoopConcurrent。 workLoopConcurrent 开始工作以后,每次协调 fiber node 时,都会判断当前时间片是否到期。如果时间片到期,会停止当前 workLoopConcurrent,让出主线程,然后请求下一个时间片继续协调。
协调的中断及恢复,类似于浏览器的eventloop,js引擎和渲染引擎互斥,在主线程中交替工作。
我们可以通过模拟 eventLoop来实现时间分片以及重新请求时间片。一段 js 程序,如果在规定时间内没有结束,那我们可以主动结束它,然后请求一个新的时间片,在下一个时间片内继续处理上一次没有结束的任务。
let taskQueue = []; // 任务列表
let shouldTimeEnd = 5ms; // 一个时间片定义为 5ms
let channel = new MessageChannel(); // 创建一个 MessageChannel 实例
function wookLoop() {
let beginTime = performance.now(); // 记录开始时间
while(true) { // 循环处理 taskQueue 中的任务
let currentTime = performance.now(); // 记录下一个任务开始时的时间
if (currentTime - beginTime >= shouldTimeEnd) break; // 时间片已经到期,结束任务处理
processTask(); // 时间片没有到期,继续处理任务
}
if (taskQueue.length) { // 时间片到期,通过调用 postMessage,请求下一个时间片
channel.port2.postMessage(null);
}
}
channel.port1.onmessage = wookLoop; // 在下一个时间片内继续处理任务
workLoop();
和浏览器的消息队列 一样, react 也会维护一个任务队列 taskQueue,然后通过 workLoop 遍历 taskQueue,依次处理 taskQueue 中的任务。
taskQueue 中收集任务是有先后处理顺序的,workLoop 每次处理 taskQueue 中的任务时,都会挑选优先级最高的任务进行处理。
每触发一次 react 更新,意味着一次 fiber tree 的协调,但协调并不会在更新触发时立刻同步进行。相反,react 会为这一次更新,生成一个 task,并添加到 taskQueue 中,fiber tree 的协调方法会作为新建 task 的 callback。当 wookLoop 开始处理该 task 时,才会触发 task 的 callback,开始 fiber tree 的协调。
任务的优先级
react在内部定义了 5 种类型的优先级,以及对应的超时时间timeout
ImmediatePriority, 直接优先级,对应用户的click、input、focus等操作;timeout为 -1,表示任务要尽快处理;UserBlockingPriority,用户阻塞优先级,对应用户的mousemove、scroll等操作;timeout为 250 ms;NormalPriority,普通优先级,对应网络请求、useTransition等操作;timeout为 5000 ms;LowPriority,低优先级(未找到应用场景);timeout为 10000 ms;IdlePriority,空闲优先级,如OffScreen;timeout为 1073741823 ms;
5 种优先级的顺序为: ImmediatePriority > UserBlockingPriority > NormalPriority > LowPriority > IdlePriority。
在确定了任务的优先级以后,react 会根据优先级为任务计算一个过期时间 expirationTime,即 expirationTime = currentTime + timeout,然后根据 expirationTime 时间来决定任务处理的先后顺序。
expirationTime越小的任务会被排在task队列的越前面,之所以需要timeout,而不是直接对比优先级等级,是为了避免低优先级任务长时间被 插队而导致一直无响应;同时,在时间分片到期时,需要根据expirationTime判断下一个要处理的任务是否过期,如果已过期,就不能让出主线程,需要立即处理。
⚠️注:react17中用Lanes重构了优先级算法,此处不展开陈述,有兴趣的同学可查阅相关文档。
获取最先处理的task
react 采用了小顶堆来存储task,实现最小优先队列,即 taskQueue 是一个小顶堆,放在堆顶的task是需要最先处理的。
使用最小堆时,有三个操作:push、pop、peek。
push,入堆操作,即将 task 添加到 taskQueue 中。添加一个新创建的 task 时,会将 task 添加到最小堆的堆底,然后对最小堆做自底向上的调整。调整时,会比较堆节点(task) 的 expirationTime,将 expirationTime 较小的 task 向上调整。* peek,获取堆顶元素,即获取需要最先处理的 task,执行 task 的 callback,开始 fiber tree 的协调。* pop,堆顶元素出堆,即 task 处理完毕,从 taskQueue 中移除。移除堆顶元素以后,会对最小堆做自顶向下的调整。调整时,也是比较堆节点(task) 的 expirationTime,将 expirationTime 较大的 task 向下调整。### 高优先级的更新中断低优先级的更新
Concurrent 模式下,如果在低优先级更新的协调过程中,有高优先级更新进来,那么高优先级更新会中断低优先级更新的协调过程。
每次拿到新的时间片以后,workLoopConcurrent 都会判断本次协调对应的优先级和上一次时间片到期中断的协调的优先级是否一样。如果一样,说明没有更高优先级的更新产生,可以继续上次未完成的协调;如果不一样,说明有更高优先级的更新进来,此时要清空之前已开始的协调过程,从根节点开始重新协调。等高优先级更新处理完成以后,再次从根节点开始处理低优先级更新。
Reconcile
前面说到,reconcile(协调)就是fiber tree 结构的更新,那么具体是怎样更新的呢?本小节就来解答这个问题。
前置知识
从jsx到dom

Step1: 从jsx生成react element:
jsx 模板通过 babel 编译为 createElement 方法;执行组件方法,触发 createElement 的执行,返回 react element;
Step2: 从react element生成fiber tree:
fiber tree中存在三种类型的指针child、sibling、return。其中,child指向第一个子节点,sibling指向兄弟节点,return指针指向父节点;*fiber tree采用的深度优先遍历,如果节点有子节点,先遍历子节点;子节点遍历结束以后,再遍历兄弟节点;没有子节点、兄弟节点,就返回父节点,遍历父节点的兄弟节点;* 当节点的return指针返回null时,fiber tree的遍历结束;Step3:fiber tree生成之后,从fiber tree到真实dom,就是处理fiber tree上对应的副作用,包括:- 所有
dom节点的新增; componentDidMount、useEffect的callback函数的触发;ref引用的初始化;
双缓存fiber tree
react 做更新处理时,会同时存在两颗 fiber tree。一颗是已经存在的 old fiber tree,对应当前屏幕显示的内容,称为 current fiber tree;另外一颗是更新过程中构建的 new fiber tree,称为 workInProgress fiber tree。
current fiber tree和workInProgress fiber tree可以通过alternate指针互相访问
当更新完成以后,使用 workInProgress fiber tree 替换掉 current fiber tree,作为下一次更新的 current fiber tree。
协调的过程
协调过程中主要做三件事情:
1.为 workInProgress fiber tree 生成 fiber node;
2.为发生变化的 fiber node标记副作用 effect;
3.收集带 effect 的 fiber node;
生成workInProgress fiber tree
workInProgress fiber tree 作为一颗新树,生成 fiber node 的方式有三种:
- 克隆(浅拷贝)
current fiber node,意味着原来的dom节点可以复用,只需要更新dom节点的属性,或者移动dom节点; - 新建一个
fiber node,意味着需要新增加一个dom节点; - 直接复用
current fiber node,表示对应的dom节点完全不用做任何处理;
复用的场景:当子组件的渲染方法(类组件的 render、函数组件方法)没有触发,(比如使用了React.memo),没有返回新的 react element,子节点就可以直接复用 current fiber node。
在日常开发过程中,我们可以通过合理使用 ShouldComponentUpdate、React.memo,阻止不必要的组件重新 render,通过直接复用 current fiber node,加快 workInProgress fiber tree 的协调,达到优化的目的。
相反,只要组件的渲染方法被触发,返回新的 react element,那么就需要根据新的 react element 为子节点创建 fiber node(通过浅拷贝或新建)。
- 如果能在
current fiber tree中找到匹配节点,那么可以通过克隆(浅拷贝)current fiber node的方式来创建新的节点; - 相反,如果无法在
current fiber tree找到匹配节点,那么就需要重新创建一个新的节点;
我们常说的diff算法就是发生在这一环节。
diff算法比较的双方是 workInProgress fiber tree 中用于构建 fiber node 的 react element 和 current fiber tree 中的 fiber node,比较两者的 key 和 type,根据比较结果来决定如何为 workInProgress fiber tree 创建 fiber node。
【 key 和 type 】:
key就是 jsx 模板中元素上的 key 属性。如果不写默认为undefined。jsx 模板转化为 react element 后,元素的 key 属性会作为 react element 的 key 属性。同样的,react element 转化为 fiber node 以后,react element 的 key 属性也会作为 fiber node 的 key 属性。
jsx 中不同的元素类型,有不同的type:
<Component name="xxxx" /> //type = Component, 是一个函数 <div></div> // type = "div", 是一个字符串 <React.Fragment></React.Fragment> // type = React.Fragment, 是一个数字(react 内部定义的);
jsx 模板转化为 react element 以后,react element 的 type 属性会根据 jsx 元素的类型赋不同的值,可能是组件函数,也可能是 dom 标签字符串,还可能是数字。 react element 转化为 fiber node 以后,react element 的 type 属性也会作为 fiber node 的 type 属性。
综上,判断拷贝 current fiber node 的逻辑,概括来就是:
reactElement.key === currentFiberNode.key && reactElement.type === currentFiberNode.type, current fiber node //可以克隆; reactElement.key !== currentFiberNode.key, current fiber node //不可克隆; reactElement.key === currentFiberNode.key && reactElement.type !== currentFiberNode.type, current fiber node //不可克隆;
diff 算法:
- 已匹配的父节点的直接子节点进行比较,不跨父节点比较;
- 通过比较
key、type来判断是否需要克隆current fiber node。只有key和type都相等,才克隆current fiber node作为新的节点,否则就需要新建一个节点。key值和节点类型type,key的优先级更高。如果key值不相同,那么节点不可克隆。 - 当比较
single react element和current fiber node list时,只需要遍历current fiber node list,比较每个current fiber node和react element的key值和type。只有key和type都相等,react element和current fiber node才能匹配。如果有匹配的,直接克隆current fiber node,作为react element对应的workInProgress fiber node。如果没有匹配的current fiber node,就需要为react element重新创建一个新的fiber node作为workInProgress fiber node。 - 当比较
react element list和current fiber node list时,还需要通过列表下标index判断wokrInProgress fiber node是否相对于克隆的current fiber node发生了移动。这也是diff中最复杂的地方。
为发生变化的fiber node标记effect
判断节点是否发生变化
- 节点只要是重新创建的而不是克隆自
current fiber node,那么节点就百分之百发生了变化,需要更新;* 节点克隆自current fiber node,需要比较props是否发生了变化,如果props发生了变化,节点需要更新;* 节点克隆自current fiber node,且是组件类型,还需要比较state是否发生了变化,如果state发生了变化,节点需要更新;常见的effect类型: Placement,放置,只针对dom类型的fiber node,表示节点需要做移动或者添加操作。Update,更新,针对所有类型的fiber node,表示fiber node需要做更新操作。PlacementAndUpdate,放置并更新,只针对dom类型的fiber node,表示节点发生了移动且props发生了变化。Ref,表示节点存在ref,需要初始化 / 更新ref.current。Deletion,删除,针对所有类型的fiber node,表示fiber node需要移除。Snapshot,快照,主要是针对类组件fiber node。当类组件fiber node发生了mount或者update操作,且定义了getSnapshotBeforeUpdate方法,就会标记Snapshot。Passive,主要针对函数组件fiber node,表示函数组件使用了useEffect。当函数组件节点发生mount或者update操作,且使用了useEffect hook,就会给fiber node标记Passive。Layout,主要针对函数组件fiber node,表示函数组件使用了useLayoutEffect。当函数组件节点发生mount或者update操作,且使用了useLayoutEffect hook,就会给fiber node标记Layout。
react 使用二进制数来声明 effect,如 Placement 为 2 (0000 0010),Update 为 4 (0000 0100)。一个 fiber node 可同时标记多个 effect,如函数组件 props 发生变化且使用了 useEffect hook,那么就可以使用 Placement | Update = 516(位运算符) 来标记。
收集带effect的fiber node
如果一个 fiber node 被标记了 effect,那么 react 就会在这个 fiber node 完成协调以后,将这个 fiber node 收集到effectList中。当整颗 fiber tree 完成协调以后,所有被标记 effect 的 fiber node 都被收集到一起。
收集fiber node的 effectList 采用单链表结构存储,firstEffect 指向第一个标记 effect 的 fiber node,lastEffect 标记最后一个 fiber node,节点之间通过 nextEffect 指针连接。
由于 fiber tree 协调时采用的顺序是深度优先,协调完成的顺序是子节点、子节点兄弟节点、父节点,所以收集带 effect 标记的 fiber node 时,顺序也是子节点、子节点兄弟节点、父节点。
Render
render也称为commit,是对协调过程中标记的effect 的处理
effect 的处理分为三个阶段,这三个阶段按照从前到后的顺序为:
1.before mutation 阶段 (dom 操作之前)
2.mutation 阶段 (dom 操作)
3.layout 阶段 (dom 操作之后)
不同的阶段,处理的 effect 种类也不相同。在每个阶段,react 都会从 effectList 链表的头部 - firstEffect 开始,按序遍历 fiber node, 直到 lastEffect。
before mutation阶段
before mutation 阶段的主要工作是处理带 Snapshot 标记的 fiber node。 从 firstEffect 开始遍历 effect 列表,如果 fiber node 带 Snapshot 标记,触发 getSnapshotBeforeUpdate 方法。
mutation阶段
mutation 阶段的主要工作是处理带 Deletion 、 Placement、PlacementAndUpdate、Update 标记的 fiber node。 在这一阶段,涉及到 dom 节点的更新、新增、移动、删除,组件节点删除导致的 componentWillUnmount、destory 方法的触发,以及删除节点引发的 ref 引用的重置。
dom 节点的更新:
- 通过原生的 API
setAttribute、removeArrribute修改dom节点的attr; - 直接修改
dom节点的style; - 直接修改
dom节点的innerHtml、textContent;
dom 节点的新增和移动:
- 如果新增(移动)的节点是父节点的最后一个子节点,那么可以直接使用
appendChild方法。 - 如果不是最后一个节点,需要使用
insertBefore方法。通过遍历找到第一个没有带Placement标记的节点作为insertBefore的定位元素。
dom节点的删除:
- 如果节点是
dom节点,通过removeChild移除; - 如果节点是组件节点,触发
componentWillUnmount、useEffect的destory方法的执行; - 如果标记
Deletion的节点的子节点中有组件节点,深度优先遍历子节点,依次触发子节点的componentWillUnmount、useEffect的destory方法的执行; - 如果标记
Deletion的节点及子节点关联了ref引用,要将ref引用置空,及ref.current=null(也是深度优先遍历);
layout 阶段
layout 阶段的主要工作是处理带 update 标记的组件节点和带 ref 标记的所有节点。 工作内容如下:
- 如果类组件节点是
mount操作,触发componentDidMount;如果是update操作,触发componentDidUpdate; - 如果函数组件节点时
mount操作,触发useLayoutEffect的callback;如果是update操作,先触发上一次更新生成的destory,再触发这一次的callback; - 异步调度函数组件的
useEffect; - 如果组件节点关联了
ref引用,要初始化ref.current;
到此这篇关于React更新渲染原理深入分析的文章就介绍到这了,更多相关React更新渲染内容请搜索好代码网以前的文章或继续浏览下面的相关文章希望大家以后多多支持好代码网!




