React中透过render函数学习(一)——workInProgress与双缓存机制

ops/2025/1/12 23:18:26/

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
    • 更新的优先级(这取决于任务的调度)
    • 更新的类型(例如,常规更新、异步更新等)

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 节点在此过程中就是那个“当前”节点,表示正在渲染或更新的部分。

  1. 当你点击按钮并触发 setState 后,React 会创建新的 workInProgress 节点,表示更新后的组件状态。
  2. React 会对比当前的 workInProgress 节点和上次渲染时的节点,找到差异部分(比如 count 更新)。
  3. 最终,React 会将差异应用到真实 DOM 上,完成渲染。

5. 返回元素

return element;
  • 最终,updateContainer 会返回传入的 element,即新的 React 元素。这只是为了保持 API 的一致性,并不会影响更新流程。

❤ 整体流程解析

  1. 接收新的元素:首先,updateContainer 接收到新的 React 元素,这可能是一个新的组件或更新后的 JSX 树。
  2. 创建更新对象:通过 createUpdate 创建一个新的更新对象。这个对象将存储新的元素以及更新的必要信息。
  3. 加入更新队列:更新对象会被加入到根节点的更新队列中,这样 React 就知道要在后续的渲染过程中处理这个更新。
  4. 调度更新scheduleUpdateOnFiber 会将这个更新任务调度到渲染队列中,等待在合适的时机进行渲染。React 会根据任务的优先级,决定何时开始执行更新操作。
  5. 返回新的元素:虽然 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 的 缓存机制 主要通过以下两个核心概念实现:

  1. 虚拟 DOM(Virtual DOM)
  2. Reconciliation(协调算法)

下面我会更详细地解释这两个概念,以及它们如何协作实现 React 的双缓存机制。

  1. 虚拟 DOM
    虚拟 DOM 是 React 用来优化渲染性能的一种技术。它是浏览器 DOM 的一个轻量级副本,用 JavaScript 对象表示,不涉及浏览器的渲染引擎。这样,React 就可以在内存中操作虚拟 DOM,而不用直接去操作真实 DOM,减少了性能开销。
  • 创建和更新虚拟 DOM:每当组件的状态(State)或属性(Props)发生变化时,React 会首先创建一个新的虚拟 DOM 树,这个虚拟 DOM 描述了组件的新状态和界面。
  • 与旧虚拟 DOM 比较:React 会把新的虚拟 DOM 和上次渲染的虚拟 DOM 进行对比,找出两者的差异部分(这个过程叫做 diff)。
  1. Reconciliation(协调算法)
    Reconciliation 是 React 中用于更新界面的算法,负责比较新旧虚拟 DOM 树,并计算出最小的变化,最终将这些变化应用到真实 DOM 上。这个过程有两个关键步骤:
  • Diffing(对比):React 会对比新旧虚拟 DOM,找出差异,生成一个“补丁”对象,包含了所有需要修改的部分。

    • Key 的作用:当渲染列表(如 map)时,React 会使用 key 属性来帮助更高效地对比虚拟 DOM。具有相同 key 的组件会被认为是相同的,React 会尽量复用已有的组件,而不是重新渲染它们。
  • Commit(应用变更):在对比之后,React 会将这些差异应用到真实 DOM 中,最终更新页面。只有实际发生变化的部分会被更新,避免了不必要的重新渲染。

缓存机制的工作流程
React 的双缓存机制涉及两个虚拟 DOM 缓存

  1. 当前渲染中的虚拟 DOM(Current Fiber)

    • 当 React 渲染时,它会为每个组件创建一个虚拟 DOM 节点,并将它们组织成一棵 Fiber 树。每个 Fiber 节点表示一个组件的状态和渲染结果。这个过程是异步的,并且会将虚拟 DOM 存储在内存中,等待后续的对比。
  2. 更新的虚拟 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 的双缓存机制是通过以下两部分来实现的:

  1. 虚拟 DOM:通过在内存中管理一份虚拟 DOM 来减少直接操作真实 DOM 的次数。
  2. Reconciliation 算法:通过对比当前的虚拟 DOM 和新的虚拟 DOM,计算出最小的差异,最后把这些差异应用到真实 DOM 中。

React 不仅在内存中保存当前和更新后的虚拟 DOM,还通过 Fiber 架构,使得渲染过程更加高效、灵活。这些机制共同作用,最终实现了 React 的双缓存机制,能够有效地提高页面渲染性能和用户体验。

workInProgress详解

Fiber 和 workInProgress

在 React 16 引入 Fiber 架构 后,React 的渲染过程变得更加细粒度化,并且支持异步渲染。通过这种架构,React 可以中断当前的渲染工作,优先处理更高优先级的任务(如用户输入、动画等),而非一次性渲染整个组件树。workInProgress 就是在这个新架构中用于表示当前渲染工作状态的一个重要数据结构。

workInProgress 的作用

  1. 表示当前渲染中的组件状态

    • 在 Fiber 架构中,workInProgress 节点是一个 Fiber 节点,它代表当前正在处理的组件。每个 Fiber 节点都描述了组件的状态、生命周期以及它的 DOM 元素等信息。
  2. 渲染过程的暂存状态

    • 在 React 渲染过程中,workInProgress 用于存储中间的渲染状态。渲染是分步进行的,workInProgress 在渲染过程中保存了当前工作单元的临时数据,比如组件的更新、状态变化等。
  3. 异步渲染的关键

    • 并发渲染(Concurrent Rendering)模式下,React 会分割渲染工作,并且允许中断和恢复渲染任务。workInProgress 就是 React 在渲染过程中用来追踪当前进度的标记。通过 workInProgress,React 知道渲染工作的哪个部分已经完成,哪个部分还需要继续进行。它使得渲染工作可以在多个阶段之间暂停和恢复。
  4. 支持 React 的“协调”过程

    • React 中的“协调”过程(Reconciliation)指的是将更新后的组件树与之前的组件树对比,找出差异并应用到 DOM 中。workInProgress 保存着当前正在协调的组件的信息,React 使用它来计算差异并决定如何更新真实 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 能够高效地进行渲染、更新,并支持异步渲染和中断任务,提供更好的用户体验。


http://www.ppmy.cn/ops/149581.html

相关文章

设计模式 创建型 抽象工厂模式(Abstract Factory)与 常见技术框架应用 解析

抽象工厂模式&#xff08;Abstract Factory Pattern&#xff09;是创建型设计模式之一&#xff0c;它提供了一种创建一系列相关或相互依赖对象的接口&#xff0c;而无需指定它们具体的类。这种模式强调了族&#xff08;family&#xff09;的概念&#xff0c;即一组具有相同主题…

Go语言如何实现高性能缓存服务

在Go语言中实现高性能缓存服务&#xff0c;需要综合考虑数据结构的选择、并发控制、内存管理以及持久化策略等多个方面。以下是一些关键步骤和最佳实践&#xff0c;可以帮助你构建高性能的缓存服务&#xff1a; 选择合适的数据结构&#xff1a; 使用哈希表&#xff08;如Go的m…

利用 Python 爬虫从义乌购根据关键词获取商品列表

在当今数字化商业时代&#xff0c;数据是企业获取竞争优势的关键。对于从事国际贸易的商家而言&#xff0c;能够及时、准确地获取商品信息至关重要。义乌购作为知名的国际贸易批发平台&#xff0c;汇集了海量的商品资源。通过 Python 爬虫技术&#xff0c;我们可以高效地从义乌…

硬件设计-齐纳管

目录 摘要 详情 齐纳管的工作电流、 摘要 齐纳管&#xff08;Zener Diode&#xff09;是一种特殊的二极管&#xff0c;它能够在特定的反向电压下保持电流稳定。正常情况下&#xff0c;二极管只允许正向电流通过&#xff0c;而阻止反向电流流过。而齐纳管在一定的反向电压下可…

WebGIS在应急灾害中对村庄、风景区、机场的影响范围应用-以日喀则市定日县地震为例

目录 前言 一、关于影响范围 1、震中距离5公里 2、震中20公里范围 3、20到80公里范围 二、空间查询知识 1、相关数据介绍 2、空间数据查询 三、前后端数据查询以及web可视化实现 1、后台API实现 2、WebGIS前端实现 四、Web成果展示 1、空间位置分析 2、包含风景区…

六、智能体强化学习——PyMARL框架

一、PyMARL 简介 PyMARL&#xff08;PyTorch Multi-Agent Reinforcement Learning&#xff09;是一个来自 QMIX 论文&#xff08;作者为 DeepMind & Oxford 合作团队&#xff09;所开源的多智能体强化学习框架。它主要面向 StarCraft Multi-Agent Challenge (SMAC) 等复杂…

实现Android应用开机自启功能

在开发某些类型的Android应用程序时&#xff0c;可能需要在设备启动后自动运行该应用。例如&#xff0c;对于企业级应用、监控软件或特定的工具类应用来说&#xff0c;这一特性尤为重要。本文将详细介绍如何通过修改AndroidManifest.xml文件并编写相应的广播接收器来实现这一目…

client-go 的 QPS 和 Burst 限速

1. 什么是 QPS 和 Burst &#xff1f; 在 kubernetes client-go 中&#xff0c;QPS 和 Burst 是用于控制客户端与 Kubernetes API 交互速率的两个关键参数&#xff1a; QPS (Queries Per Second) 定义&#xff1a;表示每秒允许发送的请求数量&#xff0c;即限速器的平滑速率…