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

相关推荐
鹏多多2 小时前
基于Vue3+TS的自定义指令开发与业务场景应用
前端·javascript·vue.js
江城开朗的豌豆2 小时前
Redux 与 MobX:我的状态管理选择心路
前端·javascript·react.js
Cosolar2 小时前
前端如何实现VAD说话检测?
前端
绝无仅有2 小时前
某辅导教育大厂真实面试过程与经验总结
后端·面试·架构
绝无仅有2 小时前
Java后端技术面试:银行业技术架构相关问题解答
后端·面试·github
CodeSheep2 小时前
当了leader才发现,大厂最想裁掉的,不是上班总迟到的,也不是下班搞失联的,而是经常把这3句话挂在嘴边的
前端·后端·程序员
吃饺子不吃馅2 小时前
✨ 你知道吗?SVG 里藏了一个「任意门」——它就是 foreignObject! 🚪💫
前端·javascript·面试
IT_陈寒3 小时前
Python开发者必须掌握的12个高效数据处理技巧,用过都说香!
前端·人工智能·后端
gnip10 小时前
企业级配置式表单组件封装
前端·javascript·vue.js