root.render(<App />)之后 React 干了哪些事?

root.render(<App />) 之后 React 到底干了哪些事、屏幕上的像素是怎么出来的、以及以后 JSX 一改页面又是如何"自动"刷新的?

一、第一次 render 之后发生了什么

  1. 创建 Fiber 根节点
    createRoot(domContainer) 会在内部生成一个 FiberRoot 对象,它记住"真实 DOM 入口"和"React 世界"的对应关系。

  2. 进入 concurrent 调度循环
    root.render(<App />) 并不会立即把 DOM 写出来,而是向 Scheduler(调度器)提交一个"更新任务"(update)。

    Scheduler 依据浏览器每一帧的空闲时间(requestIdleCallback / MessageChannel polyfill)决定"现在能不能干一点活",从而保证 60 fps 不卡帧。

  3. 生成/对比 Fiber 树(render 阶段,可中断)

    • 从根 Fiber 开始,React 深度优先遍历新返回的 JSX 元素,为每个组件/节点创建或复用对应的 Fiber 节点
    • 这个阶段完全不碰真实 DOM ,只做两件事:
      ① 调用函数组件(或 class 组件的 render)拿到 JSX → 生成子 Fiber;
      ② 在遍历过程中把"需要做的 DOM 操作"记录成一条 effectList(也叫"副作用链")。
    • 因为可中断,如果浏览器忽然要处理用户输入,React 可以把工作切片保存,先让出线程,稍后再继续。
  4. 提交阶段(commit,不可中断)

    当整棵 Fiber 树都 diff 完,React 进入同步的 commit:

    a) before mutation → 调用 getSnapshotBeforeUpdate(class 组件遗留)。

    b) mutation → 真正操作 DOM:新增、删除、移动节点,设置 textContentstyle、事件监听器......

    c) layout → 调用 useLayoutEffect / componentDidMount / componentDidUpdate;此时 DOM 已经落地,但浏览器还没 paint,适合测量 DOM 尺寸。

    d) 浏览器执行 paint ,像素出现在屏幕。

    e) passive → 异步调度 useEffect 回调(下一帧后执行)。

二、为什么能看到页面

commit 阶段的 mutation 一步直接把 DOM 增删改做完,浏览器紧跟其后执行重排/重绘,于是用户就看见了初次界面。

三、后续 JSX 改动如何更新视图

  1. 触发更新

    • 调用 setState / useState 的 setter / forceUpdate / root.render 新 JSX ... 都会生成一个"更新对象",追加到对应 Fiber 节点的队列里。
  2. 再次走"调度 → render(diff)→ commit"循环

    • 调度器依旧按优先级(用户阻塞级、普通、空闲、离线)安排任务。

    • render 阶段把"新 JSX"与"上次留下的 Fiber 树"做 对比(diff)

    • 产生的 effectList 里这次可能只有"更新某文本节点""给某个 div 加新 class"等少量操作。

    • commit 阶段再次同步执行这些 DOM 操作 → 浏览器 paint → 用户看到刷新后界面。

  3. 函数组件的"重新执行"

    函数组件在每一次 render 阶段都会被整体重新调用一次 ,Hooks 靠存在 Fiber 节点的 memorizedState 链表来保持状态,因此你写在函数体里的代码每次都会跑,但状态不会丢失。

相关推荐
Emrys_5 小时前
Redis 为什么这么快?一次彻底搞懂背-后的秘密 🚀
后端·面试
晒太阳5795 小时前
懒加载与按需加载
前端
10年前端老司机5 小时前
面试官爱问的 Object.defineProperty,90%的人倒在这些细节上!
前端·javascript
庞囧5 小时前
从输入 URL 到开始解析 HTML 之间:浏览器背后发生了什么
前端
少年阿闯~~5 小时前
解决HTML塌陷的方法
前端·html
聪明的笨猪猪5 小时前
Java Spring “事务” 面试清单(含超通俗生活案例与深度理解)
java·经验分享·笔记·面试
zycoder.5 小时前
力扣面试经典150题day3第五题(lc69),第六题(lc189)
算法·leetcode·面试
徐小夕5 小时前
花了4个月时间,我写了一款支持AI的协同Word文档编辑器
前端·vue.js·后端
岁月向前6 小时前
小组件获取主App数据的几种方案
前端
用户47949283569156 小时前
TypeScript 和 JavaScript 的 'use strict' 有啥不同
前端·javascript·typescript