React中透过render函数学习(一)——workInProgress与双缓存机制
React 18 中 updateContainer
方法的简化实现,其中包含了一些重要的操作,如创建更新对象、将更新任务加入更新队列、调度更新等。这一过程体现了 React 内部如何协调渲染过程,尤其是如何在 Fiber 架构下处理更新。让我们逐步分析这个方法的工作原理。
updateContainer
方法的实现分析
export function updateContainer(
element: ReactElementType | null,
root: FiberRootNode
) {
// 获取根 Fiber 节点
const hostRootFiber = root.current;
// 创建更新对象,包含新的 React 元素
//就是返回传入的element <App/>
const update = createUpdate<ReactElementType | null>(element);
// 将更新对象加入更新队列
enqueueUpdate(
hostRootFiber.updateQueue as UpdateQueue<ReactElementType>,
update
);
// 调度更新任务
scheduleUpdateOnFiber(hostRootFiber);
// 返回传入的 element(React 元素)
return element;
}
1. 获取根 Fiber 节点
const hostRootFiber = root.current;
root
是通过ReactDOM.createRoot()
创建的根节点对象,它表示根组件的 Fiber 树。root.current
获取的是 FiberRootNode,即 Fiber 树的根节点。hostRootFiber
是根节点的 Fiber 节点,它是渲染树的顶层节点,负责协调该树的更新。
2. 创建更新对象——createUpdate
const update = createUpdate<ReactElementType | null>(element);
createUpdate
函数创建了一个新的 更新对象,该对象表示即将更新的 React 元素。这里,element
是传入的新的 React 元素(可以是一个组件、JSX、或者null
)。createUpdate
通常会包含以下内容:- 新的 React 元素(即
element
) - 更新的优先级(这取决于任务的调度)
- 更新的类型(例如,常规更新、异步更新等)
- 新的 React 元素(即
3. 将更新加入更新队列——enqueueUpdate
enqueueUpdate(
hostRootFiber.updateQueue as UpdateQueue<ReactElementType>,
update
);
enqueueUpdate
将创建的update
对象加入到 更新队列 中。hostRootFiber.updateQueue
是 Fiber 树中根节点对应的更新队列,update
就是包含新元素信息的更新任务。- 更新队列是一个用于管理和调度更新任务的结构,React 将所有的更新任务都先存放在更新队列中,随后进行批量处理。
4. 调度更新任务——scheduleUpdateOnFiber
scheduleUpdateOnFiber(hostRootFiber);
//具体代码
function scheduleUpdateOnFiber(fiber: Fiber) {
// 步骤 1: 根据 Fiber 节点生成更新
const root = makeUpdateFromFiberToRoot(fiber);
// 步骤 2: 触发渲染过程
renderRoot(root);
}
4.1 向上查找根节点——makeUpdateFromFiberToRoot
其中makeUpdateFromFiberToRoot方法递归得到顶层节点。它负责从当前的 Fiber 节点递归地向上走,直到到达根节点。通过这种递归的方式,React 将更新从单一的组件(Fiber 节点)传递到整个渲染树。
function makeUpdateFromFiberToRoot(fiber: Fiber) {
// 步骤 1: 从 Fiber 节点找到根节点
let node = fiber;
while (node.return !== null) {
node = node.return;
}
// `node` 现在是根节点,根节点也可以称作 FiberRoot 节点
const root = node.stateNode;
// 创建并返回更新对象
return root;
}
解析:
向上查找根节点
:makeUpdateFromFiberToRoot 会从当前的 fiber 节点开始,沿着 fiber.return 链向上查找,直到找到根节点(FiberRootNode)。
生成更新对象
:一旦到达根节点,React 会生成一个更新对象,代表当前的更新任务(例如,某个组件的状态变化或新的 JSX 渲染)。
4.2 触发渲染过程——renderRoot
renderRoot
是一个函数,用于触发渲染工作。它通常会调用并递归地执行一系列的渲染任务,直到完成整个更新过程。这个过程包括执行 Fiber 树中的 beginWork 和 completeWork,以及渲染新状态或 DOM。
双缓存机制
React 使用两棵 Fiber 树
(current (上次)和 workInProgress(当前)+虚拟DOM对比))来实现双缓存
function renderRoot(root: FiberRootNode) {
//双缓存机制,将current复制一层给workInProgress
//React 使用两棵 Fiber 树(current 和 workInProgress)来实现双缓存
const { current } = root; // 获取当前的 Fiber 树的根节点
let workInProgress = current;
// 启动渲染任务(并发渲染模式下会启动任务)
workInProgress = performUnitOfWork(workInProgress);
// 继续调度工作单元
while (workInProgress !== null) {
workInProgress = performUnitOfWork(workInProgress);
}
}
此prepareFreshStatck中的createWorkInProgress简单理解就是上面将current复制一层给workInProgress, const { current } = root; // 获取当前的 Fiber 树的根节点 let workInProgress = current;
,两棵 Fiber 树来实现双缓存
。
复制过后的双缓存 数据结构
workInProgress
在 React 中,workInProgress
是与 Fiber 架构 密切相关的一个重要概念。它是 React 用来管理和跟踪组件渲染过程中的工作单元(work units)的一种数据结构。简单来说,workInProgress
代表的是 当前正在进行渲染工作 的部分。
workInProgress的数据结构
workInProgress
是一个 Fiber
对象,它包含了很多信息,比如:
tag
:当前组件的类型(比如函数组件、类组件、原生 DOM 元素等)。stateNode
:组件的实例或者 DOM 节点。pendingProps
:组件当前的 props,用于对比。memoizedProps
:上一次渲染时的 props。updateQueue
:包含了当前组件的所有更新信息。return
:当前节点的父节点,指向 Fiber 树中的父组件。
当 React 渲染时,它会不断地遍历这些 Fiber 节点,更新 workInProgress
中的状态,直到完成整个组件树的更新。
举个例子
假设有一个简单的组件树:
class App extends React.Component {
state = { count: 0 };
render() {
return (
<div>
<p>{this.state.count}</p>
<button onClick={() => this.setState({ count: this.state.count + 1 })}>
Increment
</button>
</div>
);
}
}
在 React 渲染时,React 会先构建一颗虚拟 DOM 树(Fiber 树),每个组件都有对应的 Fiber 节点。workInProgress
节点在此过程中就是那个“当前”节点,表示正在渲染或更新的部分。
- 当你点击按钮并触发
setState
后,React 会创建新的workInProgress
节点,表示更新后的组件状态。 - React 会对比当前的
workInProgress
节点和上次渲染时的节点,找到差异部分(比如count
更新)。 - 最终,React 会将差异应用到真实 DOM 上,完成渲染。
5. 返回元素
return element;
- 最终,
updateContainer
会返回传入的element
,即新的 React 元素。这只是为了保持 API 的一致性,并不会影响更新流程。
❤ 整体流程解析
- 接收新的元素:首先,
updateContainer
接收到新的 React 元素,这可能是一个新的组件或更新后的 JSX 树。 - 创建更新对象:通过
createUpdate
创建一个新的更新对象。这个对象将存储新的元素以及更新的必要信息。 - 加入更新队列:更新对象会被加入到根节点的更新队列中,这样 React 就知道要在后续的渲染过程中处理这个更新。
- 调度更新:
scheduleUpdateOnFiber
会将这个更新任务调度到渲染队列中,等待在合适的时机进行渲染。React 会根据任务的优先级,决定何时开始执行更新操作。 - 返回新的元素:虽然
updateContainer
最终返回了新的元素,但是这只是为了 API 一致性,并不会影响实际的更新过程。
❤ Fiber 树与并发渲染
在 React 18 中,Fiber 架构允许 React 在更新过程中分片任务,并在空闲时执行低优先级任务。通过 Fiber,React 能够在不阻塞主线程的情况下处理复杂的 UI 更新。
- Fiber 是 React 内部用于协调渲染的结构,表示了整个组件树的每个单独的部分。
- 更新队列 管理着所有的更新任务,每个任务都有一个 优先级,React 会根据优先级决定任务的执行顺序。
- 调度器(Scheduler) 决定任务何时执行,React 18 引入了更细粒度的调度,使得 UI 更新更加高效,避免了长时间的 UI 卡顿。
❤ 关键点总结
updateContainer
是一个用于调度更新的核心函数,它通过创建更新对象、加入更新队列并调度更新任务来管理渲染更新。createUpdate
创建一个新的更新任务,其中包含了新的 React 元素。enqueueUpdate
将更新对象加入到更新队列。scheduleUpdateOnFiber
将更新任务调度到 Fiber 渲染队列中,执行时会根据任务的优先级进行调度。- React 18 引入了 并发渲染,通过任务优先级和调度器,能够有效地在后台渲染和调度 UI 更新,避免 UI 卡顿,提升响应速度。
通过这样的机制,React 能够以更加高效的方式管理和渲染 UI,尤其适用于复杂的和交互频繁的应用。
双缓存机制
React 的 双缓存机制 主要通过以下两个核心概念实现:
- 虚拟 DOM(Virtual DOM)
- Reconciliation(协调算法)
下面我会更详细地解释这两个概念,以及它们如何协作实现 React 的双缓存机制。
- 虚拟 DOM
虚拟 DOM 是 React 用来优化渲染性能的一种技术。它是浏览器 DOM 的一个轻量级副本,用 JavaScript 对象表示,不涉及浏览器的渲染引擎。这样,React 就可以在内存中操作虚拟 DOM,而不用直接去操作真实 DOM,减少了性能开销。
- 创建和更新虚拟 DOM:每当组件的状态(State)或属性(Props)发生变化时,React 会首先创建一个新的虚拟 DOM 树,这个虚拟 DOM 描述了组件的新状态和界面。
- 与旧虚拟 DOM 比较:React 会把新的虚拟 DOM 和上次渲染的虚拟 DOM 进行对比,找出两者的差异部分(这个过程叫做 diff)。
- Reconciliation(协调算法)
Reconciliation 是 React 中用于更新界面的算法,负责比较新旧虚拟 DOM 树,并计算出最小的变化,最终将这些变化应用到真实 DOM 上。这个过程有两个关键步骤:
-
Diffing(对比):React 会对比新旧虚拟 DOM,找出差异,生成一个“补丁”对象,包含了所有需要修改的部分。
- Key 的作用:当渲染列表(如
map
)时,React 会使用key
属性来帮助更高效地对比虚拟 DOM。具有相同key
的组件会被认为是相同的,React 会尽量复用已有的组件,而不是重新渲染它们。
- Key 的作用:当渲染列表(如
-
Commit(应用变更):在对比之后,React 会将这些差异应用到真实 DOM 中,最终更新页面。只有实际发生变化的部分会被更新,避免了不必要的重新渲染。
双缓存机制的工作流程
React 的双缓存机制涉及两个虚拟 DOM 缓存:
-
当前渲染中的虚拟 DOM(Current Fiber)
- 当 React 渲染时,它会为每个组件创建一个虚拟 DOM 节点,并将它们组织成一棵 Fiber 树。每个 Fiber 节点表示一个组件的状态和渲染结果。这个过程是异步的,并且会将虚拟 DOM 存储在内存中,等待后续的对比。
-
更新的虚拟 DOM(Next Fiber)
- 在每次状态更新时,React 会创建一个新的虚拟 DOM 树,它代表了组件状态更新后的视图。这个新的虚拟 DOM 会和当前虚拟 DOM 进行对比(通过 Reconciliation),找出差异。
这个新的虚拟 DOM 树可以看作是 下一次渲染的候选树,而 当前虚拟 DOM 是正在显示在页面上的。
具体步骤
- 更新触发:当组件的状态发生变化时,React 会创建一个新的虚拟 DOM 树。
- 比较新旧虚拟 DOM:React 使用 Diff 算法对比当前的虚拟 DOM 和新的虚拟 DOM,找出它们之间的差异。
- 应用差异:将这些差异(也称为补丁)应用到真实 DOM 上,最小化更新和渲染的开销。
Fiber架构的引入
从 React 16 开始,React 引入了 Fiber 架构,这使得 React 能够更加细粒度地控制渲染过程,从而支持更加高效的双缓存机制。
- 在 Fiber 架构下,React 将渲染过程分为多个 工作单元,并且允许中断这些工作单元,优先处理高优先级的任务(如用户交互),然后再继续处理低优先级的任务(如动画)。这种方式提高了 React 渲染的响应性,避免了“卡顿”的现象。
总结
React 的双缓存机制是通过以下两部分来实现的:
- 虚拟 DOM:通过在内存中管理一份虚拟 DOM 来减少直接操作真实 DOM 的次数。
- Reconciliation 算法:通过对比当前的虚拟 DOM 和新的虚拟 DOM,计算出最小的差异,最后把这些差异应用到真实 DOM 中。
React 不仅在内存中保存当前和更新后的虚拟 DOM,还通过 Fiber 架构,使得渲染过程更加高效、灵活。这些机制共同作用,最终实现了 React 的双缓存机制,能够有效地提高页面渲染性能和用户体验。
workInProgress详解
Fiber 和 workInProgress
在 React 16 引入 Fiber 架构 后,React 的渲染过程变得更加细粒度化,并且支持异步渲染。通过这种架构,React 可以中断当前的渲染工作,优先处理更高优先级的任务(如用户输入、动画等),而非一次性渲染整个组件树。workInProgress
就是在这个新架构中用于表示当前渲染工作状态的一个重要数据结构。
workInProgress 的作用
-
表示当前渲染中的组件状态
- 在 Fiber 架构中,
workInProgress
节点是一个 Fiber 节点,它代表当前正在处理的组件。每个 Fiber 节点都描述了组件的状态、生命周期以及它的 DOM 元素等信息。
- 在 Fiber 架构中,
-
渲染过程的暂存状态
- 在 React 渲染过程中,
workInProgress
用于存储中间的渲染状态。渲染是分步进行的,workInProgress
在渲染过程中保存了当前工作单元的临时数据,比如组件的更新、状态变化等。
- 在 React 渲染过程中,
-
异步渲染的关键
- 在 并发渲染(Concurrent Rendering)模式下,React 会分割渲染工作,并且允许中断和恢复渲染任务。
workInProgress
就是 React 在渲染过程中用来追踪当前进度的标记。通过workInProgress
,React 知道渲染工作的哪个部分已经完成,哪个部分还需要继续进行。它使得渲染工作可以在多个阶段之间暂停和恢复。
- 在 并发渲染(Concurrent Rendering)模式下,React 会分割渲染工作,并且允许中断和恢复渲染任务。
-
支持 React 的“协调”过程
- React 中的“协调”过程(Reconciliation)指的是将更新后的组件树与之前的组件树对比,找出差异并应用到 DOM 中。
workInProgress
保存着当前正在协调的组件的信息,React 使用它来计算差异并决定如何更新真实 DOM。
- React 中的“协调”过程(Reconciliation)指的是将更新后的组件树与之前的组件树对比,找出差异并应用到 DOM 中。
workInProgress 和 Fiber 树
在 Fiber 架构中,React 渲染流程实际上是围绕一颗 Fiber 树 进行的。每个组件、元素或节点都会有一个对应的 Fiber 节点。workInProgress
就是表示渲染过程中正在执行任务的那个节点。
具体来说,React 会构建一个新的 Fiber 树来表示新的 UI,而 workInProgress
就是这个树的当前节点,它表示正在进行的工作单元。
workInProgress
与“当前”和“上次”渲染的关系
-
当前渲染(Work In Progress):这个状态表示 React 当前正在处理的工作单元,也就是正在被更新或渲染的部分。React 会把每个组件的 Fiber 节点从头到尾进行处理,生成新的状态。
-
上一次渲染(Current):这个状态表示上一次完成渲染的结果,它是一个稳定的“当前” Fiber 树,代表了浏览器上实际渲染的 UI。它是与“工作中”的
workInProgress
树进行对比的对象。
在 React 渲染的过程中,workInProgress
代表的是正在处理的虚拟 DOM 树,而 React 会根据 workInProgress
和当前的 Fiber 树(即上次渲染的结果)进行 Diff 和 Reconciliation。
总结
workInProgress
是 React Fiber 架构中的一个核心概念,表示当前正在进行的渲染工作。它保存了当前组件的渲染状态,并帮助 React 处理渲染过程中的协调和更新。通过这种方式,React 能够高效地进行渲染、更新,并支持异步渲染和中断任务,提供更好的用户体验。