React:渲染帧和事件循环是什么关系

React为减缓浏览器单帧渲染压力做了时间切片处理,然后通过宏任务进行调度,其底层基于浏览器的渲染事件循环 的原理。浏览器中的渲染帧事件循环是紧密协作但职责分明的两个核心机制,它们的关系可以通过以下要点清晰呈现:


1. 层级关系

plaintext 复制代码
事件循环(Event Loop)
├─ 宏任务队列(Task Queue)
├─ 微任务队列(Microtask Queue)
└─ 渲染帧(Frame)
   ├─ 样式计算(Style)
   ├─ 布局(Layout)
   └─ 绘制(Paint)

2. 协作流程

react/packages/scheduler/src/forks/Scheduler.js 中体现的协作逻辑:

javascript 复制代码
function performWorkUntilDeadline() {
  // 阶段1:执行宏任务(如React调度任务)
  flushWork(); 
  
  // 阶段2:清空微任务队列(浏览器内部行为)
  
  // 阶段3:执行渲染管线(浏览器内部行为)
  if(needsPaint) requestPaint(); 
}

标准事件循环迭代

  1. 从宏任务队列取出一个任务执行
  2. 执行所有微任务(直到队列清空)
  3. 执行渲染管线(如果需要)
  4. 进入下一轮循环

3. 关键差异

特性 事件循环 渲染帧
触发频率 持续运行 按屏幕刷新率触发(如60Hz)
核心职责 任务调度与执行 视觉更新计算
可中断性 不可中断(必须执行到微任务清空) 可跳过部分阶段
性能影响 长任务导致事件循环卡顿 复杂样式导致渲染耗时

4. React的协调策略

React调度器通过时间切片主动适配两者:

javascript 复制代码
function workLoop() {
  while (task && !shouldYield()) {
    process(task); // 每次执行不超过5ms
  }
  // 主动让出控制权给浏览器
  if (hasMoreWork) schedulePerformWorkUntilDeadline();
}

这样设计使得:

  1. 事件循环不会被长时间阻塞
  2. 渲染帧能获得稳定的执行机会
  3. 高优先级更新能及时反映到视图

5. 实际案例对比

原生事件循环:

javascript 复制代码
// 案例1:阻塞型微任务
Promise.resolve().then(() => {
  heavyTask(50ms); // 直接导致3帧丢失
});

React调度:

javascript 复制代码
// 案例2:时间切片
function task() {
  chunkedWork(5ms); // 分块执行
  if (remainingWork) {
    scheduleTask(task); // 下一宏任务继续
  }
}

总结关系图

plaintext 复制代码
[事件循环] ← 控制权交替 → [渲染帧]
   ↑                       ↑
   │                       │
宏任务/微任务           样式/布局/绘制
   │                       │
   └─── React调度器协调 ───┘

这种设计就像双人接力赛跑

  • 事件循环是持棒的奔跑者(必须持续前进)
  • 渲染帧是交棒检查点(按固定节奏接棒)
  • React调度器是教练(控制交接时机和节奏)

问题

问:如果React代码中有比执行较长时间的微任务呢?也能保持流畅吗?

答:当然是否定。因为React对宏任务做了时间切片,微任务在渲染之前执行,如果时间过长,仍然会阻塞渲染。

相关推荐
JinSo18 分钟前
pnpm monorepo 联调:告别 --global 参数
前端·github·代码规范
程序员码歌25 分钟前
豆包Seedream4.0深度体验:p图美化与文生图创作
android·前端·后端
urhero29 分钟前
工作事项管理小工具——HTML版
前端·html·实用工具·工作事项跟踪·任务跟踪小工具·本地小程序
二十雨辰31 分钟前
eduAi-智能体创意平台
前端·vue.js
golang学习记39 分钟前
从0死磕全栈之Next.js connection() 函数详解:强制动态渲染的正确姿势(附实战案例)
前端
郝学胜-神的一滴1 小时前
Three.js光照技术详解:为3D场景注入灵魂
开发语言·前端·javascript·3d·web3·webgl
m0dw1 小时前
vue懒加载
前端·javascript·vue.js·typescript
国家不保护废物1 小时前
手写 Vue Router,揭秘路由背后的魔法!🔮
前端·vue.js
菜鸟‍2 小时前
【前端学习】仿Deepseek官网AI聊天网站React
前端·学习·react.js
小光学长2 小时前
基于Vue的保护动物信息管理系统r7zl6b88 (程序+源码+数据库+调试部署+开发环境)带论文文档1万字以上,文末可获取,系统界面在最后面。
前端·数据库·vue.js