【React】React组件的渲染过程分为哪几个阶段?

⭐ 总体结构

复制代码
触发渲染
   ↓
🚧 1. Render 阶段(协调阶段 / Reconciliation)
   ↓
⚡️ 2. Commit 阶段(提交阶段 / DOM 操作)

⭐ 一、触发渲染的三个来源

React 不会无缘无故渲染,一个组件的渲染只会由以下情况触发:

1️⃣ 父组件渲染导致子组件渲染(默认行为)

2️⃣ 组件内部状态更新(setState / useState)

3️⃣ Context 变化导致使用 useContext 的组件更新


⭐ 二、React 渲染过程

🚧 阶段 1:Render 阶段(Reconciliation 协调阶段)

目标:计算出需要更新什么,生成新的 Fiber Tree

这是一个 纯计算阶段

  • 不会操作 DOM
  • 可被暂停、可打断(Concurrent Mode)
  • 可恢复
Render 阶段会做什么?
  1. 执行函数组件(执行组件体)
  2. 运行 Hooks(useState、useMemo 等)
  3. 比较新旧 Fiber 树(diff)
  4. 生成更新链表(effect list)
Render阶段产物:
  • 一个 需要被提交到 DOM 的更新表

注意:render 不是指"渲染到 DOM",而是"构造新的虚拟 fiber 树"。


⚡️ 阶段 2:Commit 阶段(提交阶段 / 实际 DOM 操作)

Commit 阶段无法被打断,因为涉及到真实 DOM。

分三步:

1️⃣ before mutation(提交前)

执行:

  • getSnapshotBeforeUpdate
  • 调整内容前的准备
2️⃣ mutation(变更 DOM)

执行所有 DOM 操作:

  • 插入 DOM
  • 删除 DOM
  • 修改 DOM 属性
  • ref 的更新
3️⃣ layout(布局阶段)

执行 Layout Effects:

  • useLayoutEffect 回调
  • class 组件的 componentDidMount / componentDidUpdate

Mutation 和 Layout 这两步是同步执行的,不允许中断。


⭐ React Hook 与两个阶段的关系

Hook 执行阶段
useState setter → 触发 Render 渲染开始前
useMemo / useCallback Render 阶段
useEffect Commit - layout 后(异步)
useLayoutEffect Commit - mutation 与 paint 前
ref 更新 Commit - mutation 阶段

两个关键点:

  • useEffect 在 Commit 后异步执行,不阻塞渲染
  • useLayoutEffect 会阻塞浏览器绘制(类似 componentDidMount)

React 为什么要分 Render 和 Commit?

🧠 Render 阶段可中断=流畅度提升

因为计算可以被暂停,让位给用户输入。

⚡️ Commit 阶段不可中断=保持 DOM 一致性

保证真实 UI 操作的原子性,不会出现中断状态。


⭐ 完整流程图

复制代码
state 更新 / props 更新 / context 更新
        ↓
---- Render 阶段(可中断) ----
1. 执行组件函数
2. 运行 hooks
3. 生成新的 Fiber 树
4. 计算 diff
5. 构建 effectList(DOM 变更清单)
        ↓
---- Commit 阶段(不可中断) ----
1. before mutation(DOM 操作前的准备)
2. mutation(真实 DOM 更新)
3. layout(执行 useLayoutEffect、componentDidMount)
        ↓
浏览器绘制(paint)
        ↓
最后执行 useEffect(异步)

相关推荐
程序员猫哥几秒前
前端开发,一句话生成网站
前端
Younglina21 分钟前
一个纯前端的网站集合管理工具
前端·vue.js·chrome
木头程序员22 分钟前
前端(包含HTML/JavaScript/DOM/BOM/jQuery)基础-暴力复习篇
开发语言·前端·javascript·ecmascript·es6·jquery·html5
卖火箭的小男孩24 分钟前
# Flutter Provider 状态管理完全指南
前端
小雨青年24 分钟前
鸿蒙 HarmonyOS 6|ArkUI(01):从框架认知到项目骨架
前端
Null15527 分钟前
浏览器唤起本地桌面应用(基础版)
前端·浏览器
哈__31 分钟前
React Native 鸿蒙跨平台开发:PixelRatio 实现鸿蒙端图片的高清显示
javascript·react native·react.js
wszy180940 分钟前
外部链接跳转:从 App 打开浏览器的正确姿势
java·javascript·react native·react.js·harmonyos
pas13642 分钟前
31-mini-vue 更新element的children
前端·javascript·vue.js
wordbaby1 小时前
TanStack Router 实战:如何构建经典的“左侧菜单 + 右侧内容”后台布局
前端·react.js