Suspense and Fiber: The Intricate Machinery Behind React&#s Rendering Elegance

suspense and fiber: the intricate machinery behind react

react fiber 是 react 并发渲染的核心,它使框架能够将任务分解为更小的单元,并优先处理更重要的任务,从而实现更流畅、响应更灵敏的用户界面。当与 suspense 配合使用时,它允许 react “暂停”渲染,在等待数据获取或计算等任务完成时显示后备 ui。

fiber 是一个 javascript 对象,代表 react 中的一个工作单元。它在渲染过程中保存有关组件的重要信息:

那篇文章中的所有代码都是伪代码。真实文件链接可以在文末找到

{  tag,                // the type of fiber (e.g., hostcomponent, functioncomponent)  statenode,          // the actual dom node or instance for class components  memoizedprops,      // props used during the last render  memoizedstate,      // state used during the last render  return,             // parent fiber  sibling,            // next sibling fiber  child,              // first child fiber  alternate,          // link to the previous fiber (for reconciling updates)}

纤程树 允许 react 高效地遍历组件树,执行渲染工作,并跟踪更新、状态和 dom 突变。

让我们构建一个由页眉、内容和页脚组件组成的简单应用程序。为了强调 fiber 的工作原理,我们将让 header 执行繁重的计算。

function app() {  return (    <suspense fallback={
loading...
}>
);}function header() { // simulating heavy computation for (let i = 0; i < 1e9; i++) {} return

header with heavy computation

;}function content() { return

this is the content area

;}function footer() { return
footer content
;}

它是如何开始的? render 函数负责启动 react 中的初始渲染过程。它需要一个根组件(如 app)并为 react 创建初始的 fiber 树:

function render(element, container) {  const rootfiber = {    tag: hostroot,           // root fiber tag    statenode: container,    // reference to the dom container    child: null,             // initial child will be added here  };  // create a new work-in-progress fiber for our app component  const appfiber = createfiberfromelement(element);  rootfiber.child = appfiber;  // start the rendering process  schedulework(rootfiber);}

然后工作安排就开始行动了。 schedulework 负责启动工作循环并处理 fiber 树。它决定何时开始处理工作,通常使用workloop。

function schedulework(rootfiber) {  nextunitofwork = rootfiber; // set the root fiber as the starting point  requestidlecallback(workloop); // begin processing fibers when the browser is idle}

这里,根 fiber 被设置为 nextunitofwork,react 调度 workloop 函数来处理它。 react 使用 requestidlecallback 等待浏览器空闲后再开始工作,这对于非阻塞渲染至关重要。

如前所述,workloop 和performunitofwork 处理纤程的实际处理。这是协调阶段

workloop 通过调用 performunitofwork 来处理每个 fiber,直到所有任务完成或浏览器需要优先处理其他任务。

function workloop() {  while (nextunitofwork && !shouldyield()) {    nextunitofwork = performunitofwork(nextunitofwork); // perform work for the next fiber  }  if (!nextunitofwork) {    // commit all completed work to the dom    commitroot();  } else {    // yield control and schedule the next chunk of work    requestidlecallback(workloop);  }}

performunitofwork 处理各个纤维,检查 suspense 边界,启动渲染任务并管理更新。

function performunitofwork(fiber) {  const next = beginwork(fiber);  // if there's more work to do, return the next unit  if (next) {    return next;  }  // otherwise, complete the work and move to the next sibling or parent  let sibling = fiber.sibling;  if (sibling) {    return sibling;  }  return fiber.return;}function beginwork(fiber) {  if (fiber.tag === suspensecomponent) {    if (fiber.suspensestate === suspended) {      // render fallback ui if data is not ready      return fiber.fallback;    } else if (fiber.suspensestate === resolved) {      // render actual content once data resolves      return fiber.child;    }  }  // continue rendering for other fibers}

完成所有工作单元后,commitroot 负责将 fiber 树提交到 dom,包括处理 suspense fallback ui。这是提交阶段

function commitRoot() {  let fiber = rootFiber;  while (fiber) {    if (fiber.tag === SuspenseComponent && fiber.suspenseState === Suspended) {      // Commit fallback UI if Suspense is active      updateDOM(fiber.stateNode);    } else {      // Commit normal content      updateDOM(fiber.memoizedProps);    }    // If the fiber has a passive effect (i.e., useEffect), schedule it    if (fiber.effectTag === Passive) {    // Function to schedule the useEffect hooks to run after the DOM updates and browser paint      scheduleEffectHooks(fiber);    }    fiber = fiber.next;  // Move to the next fiber in the tree  }}

在提交阶段,react 会为任何具有副作用(例如 useeffect)的 fiber 调用 scheduleeffecthooks。这些钩子被推迟到 dom 更新和浏览器绘制之后,以避免阻塞渲染过程。这使得 react 能够执行非阻塞更新,同时确保 ui 稳定后及时执行副作用。

在本文中,我们探讨了 react fiber 架构的内部工作原理以及 suspense 如何与其交互以增强渲染过程。我们了解了 react 如何将渲染任务分解为更小的单元,确定重要任务的优先级,并推迟不太重要的更新,以保持用户界面流畅且响应迅速。

我们对 fiber 树结构、工作调度过程以及协调和提交阶段如何有效更新 dom 有了清晰的了解。我们还探讨了 suspense 如何集成到该架构中,在等待数据或繁重计算时显示后备内容。

如果您对本文有任何反馈或想法 – 无论部分不清楚还是可以改进 – 我很高兴听到它们。您的见解将有助于为每个人完善和增强内容。

来源和原始实现:

反应纤维概述:反应悬念:performunitofwork 的当前实现:beginwork 的当前实现:commitroot 的当前实现:schedulework 和 workloop 的当前实现

以上就是Suspense and Fiber: The Intricate Machinery Behind React&#s Rendering Elegance的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1491496.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月19日 14:00:59
下一篇 2025年12月17日 16:16:31

相关推荐

  • 如何使用 Tailwind CSS 和 JavaScript 创建基本看板

    今天是星期一,让我们深入构建一个简单的看板!我们将使用 Tailwind CSS 和 JavaScript 创建一个包含三列的列:待办事项、进行中和完成。 什么是看板?看板是用于管理工作流程中的任务的可视化工具。它将任务组织成列,每个列代表流程的不同阶段。例如,基本的看板可以包括“待办事项”、“进行…

    2025年12月19日
    000
  • React 中的上下文:更好的方法

    首先看一下这段代码: usercontext.js import { createcontext, usecontext } from “react”;const usercontext = createcontext({ user: { name: ‘default user’, age: 0 }…

    2025年12月19日
    000
  • 掌握 Lerna:管理 JavaScript Monorepos 的指南

    目录 简介第一章:lerna 是什么?为什么选择 monorepos?第 2 章:安装和设置 lerna先决条件分步安装指南设置您的第一个 lerna 项目第 3 章:lerna 中的依赖关系管理独立依赖提升共享依赖项引导包第 4 章:跨包运行脚本全局执行脚本针对特定包第 5 章:使用 lerna …

    2025年12月19日
    000
  • 使用 Lerna 掌握 Monorepos:综合指南

    简介 管理具有多个相互依赖的包的大型项目对许多开发团队来说是一个挑战。传统方法通常涉及为每个包使用多个存储库,这可能会导致代码维护、依赖项管理和协作方面的开销。 lerna 是一款功能强大的 javascript 工具,通过引入一种有效的方法来管理 monorepos(在单个代码库中托管多个包的存储…

    2025年12月19日
    000
  • 优化 Nextjs 应用性能的经过验证的技巧 ⚡️

    优化 web 应用程序的性能对于提供快速、流畅的用户体验至关重要。 使用 next.js 这个强大的 react 框架,您可以利用许多内置功能来提高应用程序的速度和效率。 以下十个关键策略可帮助您的 next.js 应用获得最佳性能: 1. 仅加载您需要的 javascript 和 css 为了避免…

    2025年12月19日
    000
  • 在画中画窗口中安装 React 组件

    google 在 chrome 116 中引入了 documentpictureinpicture api。 在本文中,我们将探讨如何在画中画窗口中安装一个简单的 react 组件,而无需先将其安装在我们的主应用程序上。 第 1 步 – 设置组件结构我们制造两个组件。 maincompo…

    2025年12月19日
    000
  • React 心态:新 React 开发人员应该如何思考

    react 是一个用于构建用户界面的流行 javascript 库,它使开发人员能够创建可重用的组件并有效管理复杂的 ui,从而彻底改变了前端开发。然而,采用正确的心态对于新开发人员驾驭 react 独特范例至关重要。让我们探索塑造“react 心态”的基本原则和策略。 1. 组件化思考 react…

    2025年12月19日
    000
  • Ajv-ts 有什么新消息?

    架构验证工具包的最新版本 0.9 – ajv-ts 带来了一系列增强功能,以提高开发人员的工作效率并确保更好的验证准确性。无论您是构建复杂的数据模型还是仅处理基本输入,这些更新都有助于通过更强大的功能和示例简化架构定义。以下是此版本中引入的关键更新的概述,以及对版本 0.7. 的重大更改…

    2025年12月19日
    000
  • JavaScript 库是否已经失控?

    有多少个图书馆? 截至本文发表时,npm 托管了超过 200 万个包,其中很大一部分是 JavaScript 库。所以你可以说有数百万个包裹。 最近,我一直在想有多少 JavaScript 库和包,而且……很多。每次我转身,都会有一个新的,无论是状态管理、动画,甚至只是处理表单输入。几乎是压倒性的!…

    2025年12月19日
    000
  • 在深入 Web 开发之前您必须了解的事项

    1. HTML、CSS、JavaScript——三位一体 没有砖头就建不了房子,不掌握 HTML、CSS 和 JavaScript 就无法建立网站。 HTML 是您的结构,CSS 使其看起来令人惊叹,而 JavaScript 通过交互性使其栩栩如生。如果您认真对待网络开发,请从这里开始。没有捷径! …

    2025年12月19日
    000
  • 了解现代 Web 开发中的 chunkjs:代码分割和性能优化指南

    在 Web 开发中,尤其是使用 React、Vue 或 Angular 等现代 JavaScript 框架时,chunk.js 指的是在应用程序的构建过程中创建的 JavaScript 捆绑文件。 当捆绑或编译 Web 应用程序时,Webpack 或 Vite 等构建工具会将 JavaScript …

    2025年12月19日
    000
  • 事件循环如何处理微任务和宏任务

    在 javascript 中,微任务和宏任务是事件循环管理的两种类型的异步任务,但它们的处理方式不同。了解它们的工作原理对于预测异步代码的执行顺序至关重要。 1.宏任务队列(任务队列) 宏任务被放入自己的队列中,通常称为任务队列或宏任务队列。宏任务的示例包括:settimeout、setinterv…

    2025年12月19日
    000
  • 你能解决这个问题吗?

    假设我们有两个 div,一个内部 div 和另一个外部 div。 内部 div 负责缩放和平移(使用 css 变换、平移和缩放完成)。 现在可以通过拖放将新的 div 添加到内部 div 中。放置事件被outerdiv 接受 现在,在放置事件期间,您可以获得 clientx、clienty 位置,现…

    2025年12月19日
    000
  • JavaScript 数组方法:综合指南

    数组是 javascript 中最基本的数据结构之一。使用数组,您可以在单个变量中存储多个值。 javascript 提供了许多内置方法来操作数组,使它们具有令人难以置信的通用性。在这篇文章中,我们将探讨所有内置数组方法以及如何在 javascript 项目中有效地使用它们。 核心方法 foreac…

    2025年12月19日
    000
  • Web Development Job in 4

    简介 网络开发领域持续快速发展,为拥有适当技能和知识的个人提供了令人兴奋的机会。如果您想进入这个充满活力的行业或提升自己的职业生涯,这里有一些重要策略可以帮助您在 2024 年找到梦想的工作: 1. 掌握基本技能 编程语言:熟练掌握 JavaScript、Python、Ruby 或 PHP 等语言至…

    2025年12月19日
    000
  • 管理速率限制

    在当今的互联世界中,应用程序通常必须对外部服务进行数十万次 API 调用,对此类请求的有效管理至关重要。为防止滥用或过度使用资源而采用的最多产的技术之一是速率限制 – 限制客户端在给定时间段内可能发出的请求数量。虽然速率限制可以提高服务的稳定性,但这对于开发人员来说通常是一个挑战,他们必…

    2025年12月19日
    000
  • JavaScript 中的高阶函数

    什么是高阶函数? 高阶函数是可以接受其他函数作为参数或返回函数作为值的函数。它们为我们提供了一种强大的方法来编写可应用于各种场景的可重用代码。 javascript 有几个常用的高阶函数,包括map()、filter() 和reduce()。由于存在高阶函数,javascript 通常被称为函数式编…

    2025年12月19日
    000
  • 为什么 Streams API 改变了 Web 开发者的游戏规则

    我们首先解释一下数据是如何通过网络发送的。它不是作为单个连续流发送的;相反,它被分成更小的块。在接收端,消费者或应用程序负责在收到所有数据后以正确的顺序和格式重新组装这些块。对于图像、视频和其他相对较大的数据类型,此过程会自动发生。 因此 streams api 提供的是一种无需等待完整数据可用的方…

    2025年12月19日
    000
  • 学习如何:useEFFECT 和 useSTATE,一个 REACT 应用程序

    我一直在开发一个名为“Heat”的网络应用程序,该应用程序的唯一目的是帮助我学习和应用 JavaScript 并改进我的 CSS,还有很多后端。我们的计划是创建一个具有视觉吸引力的网络应用程序,并具有一些有趣的功能。 在这个过程中我学到了什么: 前端是使用react js构建的,我选择react只是…

    2025年12月19日
    000
  • Bunjs 与 PM2

    动机 我正在使用 Bunjs 作为我的多人海战游戏 Sunbaked 的后端。 现在,我想让我的服务器在抛出服务器中断错误时自动重新启动。我通读了关于如何将 PM2 与 Bunjs 一起使用的官方指南,但发现它遗漏了一小段信息…… $bun install -g pm2 文章…

    2025年12月19日
    000

发表回复

登录后才能评论
关注微信