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 链表来保持状态,因此你写在函数体里的代码每次都会跑,但状态不会丢失。

相关推荐
夏鹏今天学习了吗3 小时前
【性能优化】前端高性能优化策略
前端·性能优化
ShineWinsu6 小时前
对于数据结构:堆的超详细保姆级解析——下(堆排序以及TOP-K问题)
c语言·数据结构·c++·算法·面试·二叉树·
weixin_427771616 小时前
css font-size 的妙用
前端·css
_Power_Y6 小时前
计算机网络面试题
面试
凤凰战士芭比Q7 小时前
web中间件——Nginx
前端·nginx·中间件
一 乐7 小时前
点餐|智能点餐系统|基于java+ Springboot的动端的点餐系统小程序(源码+数据库+文档)
java·前端·数据库·vue.js·spring boot·小程序·论文
视图猿人7 小时前
RxJS基本使用及在next.js中使用的例子
开发语言·javascript
火山上的企鹅7 小时前
Qt C++ 软件开发工程师面试题
c++·qt·面试
bitbitDown8 小时前
从零打造一个 Vite 脚手架工具:比你想象的简单多了
前端·javascript·面试
沐怡旸8 小时前
【穿越Effective C++】条款16:成对使用new和delete时要采用相同形式——内存管理的精确匹配原则
c++·面试