🚀React 更新界面全流程:从 setState 到 像素上屏

#金石焕新程

🚀《React 更新界面全流程:从 setState 到 像素上屏》

面试常被追问:

"React 更新界面到底经历了哪些阶段?"

本文用一张图 + 一段可运行代码,带你走通 React 18(含并发特性) 的完整刷新链路。


📊 总览:一次更新 6 步曲

阶段 发生位置 关键 API
① 触发更新 开发者代码 setState / dispatch
② 构造更新对象 React 内部 Update
③ 调度 & 优先级 Fiber Scheduler scheduleUpdateOnFiber
④ Render 阶段(可中断) Fiber beginWorkcompleteWork
⑤ Commit 阶段(同步) DOM commitWorkcommitMutationEffects
⑥ 浏览器绘制 浏览器 requestAnimationFrame → 像素上屏

① 触发更新:setState 不是立刻刷 DOM!

jsx 复制代码
function Counter() {
  const [count, setCount] = useState(0);

  // 触发更新
  const handleClick = () => setCount(c => c + 1);

  return <button onClick={handleClick}>{count}</button>;
}
  • setCount 只是把 更新任务 放进 Fiber 队列,不会立即渲染

② 构造更新对象(Fiber 视角)

js 复制代码
// React 内部伪代码
const update = {
  lane: SyncLane,      // 优先级
  action: c => c + 1,  // 更新函数
  next: null           // 链表结构
};
fiber.updateQueue = appendUpdate(fiber.updateQueue, update);

③ 调度 & 优先级(时间切片)

js 复制代码
// React 18 并发调度
scheduleUpdateOnFiber(rootFiber, lane);
  • 高优先级任务 (用户输入)插队 ,低优先级任务让步
  • 浏览器空闲时继续可中断渲染

④ Render 阶段:Fiber 可中断遍历

plaintext 复制代码
beginWork → completeWork
  • 可中断:长列表渲染可分段进行(避免白屏)。
  • 双缓冲:一边渲染 Fiber 树,一边复用旧树。

⑤ Commit 阶段:同步提交到 DOM

plaintext 复制代码
commitWork
  ├─ commitMutationEffects  // 插入 / 更新 / 删除 DOM
  ├─ commitLayoutEffects    // 调用 useLayoutEffect
  └─ commitPassiveEffects   // 调用 useEffect
  • 同步不可中断,确保 DOM 与 React 树一致。

⑥ 浏览器绘制:像素上屏

  • React 把 styletransform 交给浏览器。
  • 浏览器在下一帧 VSync 信号后完成绘制。

🧪 实战:打断渲染(Concurrent Feature)

jsx 复制代码
import { startTransition } from 'react';

function Search() {
  const [query, setQuery] = useState('');
  const [list, setList] = useState([]);

  const handleChange = (e) => {
    const q = e.target.value;
    setQuery(q);
    startTransition(() => {
      // 低优先级更新:长列表过滤
      setList(filterList(q));
    });
  };

  return (
    <>
      <input value={query} onChange={handleChange} />
      {list.map(item => <li key={item.id}>{item.name}</li>)}
    </>
  );
}
  • startTransition 把长列表过滤标记为低优先级 ,高优先级输入不卡顿

🔍 调试技巧:浏览器 Performance 面板

  1. 火焰图:查看 Render、Commit 耗时。
  2. React DevTools Profiler:识别重渲染组件。
  3. Paint Flashing:高亮重绘区域。

🏁 一句话总结

React 更新 = 调度优先级 + 可中断渲染 + 同步提交 DOM + 浏览器绘制

记住 6 步曲,面试问「React 怎么更新界面」直接答:

"从 setState 触发 → Fiber 调度 → Commit → 像素上屏!"

相关推荐
编程彩机9 小时前
互联网大厂Java面试:从分布式缓存到消息队列的技术场景解析
java·redis·面试·kafka·消息队列·微服务架构·分布式缓存
我的xiaodoujiao9 小时前
使用 Python 语言 从 0 到 1 搭建完整 Web UI自动化测试学习系列 44--将自动化测试结果自动推送至钉钉工作群聊
前端·python·测试工具·ui·pytest
沛沛老爹9 小时前
Web开发者转型AI:多模态Agent视频分析技能开发实战
前端·人工智能·音视频
David凉宸9 小时前
vue2与vue3的差异在哪里?
前端·javascript·vue.js
笔画人生9 小时前
Cursor + 蓝耘API:用自然语言完成全栈项目开发
前端·后端
AC赳赳老秦9 小时前
外文文献精读:DeepSeek翻译并解析顶会论文核心技术要点
前端·flutter·zookeeper·自动化·rabbitmq·prometheus·deepseek
小宇的天下9 小时前
Calibre 3Dstack --每日一个命令day18【floating_trace】(3-18)
服务器·前端·数据库
毕设源码-钟学长9 小时前
【开题答辩全过程】以 基于web技术的酒店信息管理系统设计与实现-为例,包含答辩的问题和答案
前端
xiaoye-duck9 小时前
C++ string 底层原理深度解析 + 模拟实现(上)——面试 / 开发都适用
c++·面试·stl
css趣多多9 小时前
this.$watch
前端·javascript·vue.js