Vue3的渲染秘密:从同步批处理到异步微任务

面试里,当面试官把两段看似「都是改 5 次数据」的代码摆在你面前,却问「渲染了几次?」,如果你只回答「改了 5 次所以 5 次」,那大概率就踩坑了。本文用 100 行代码把同步批处理与异步微任务的底层机制拆开讲透,让你以后遇到同类问题直接秒答。

一、手写一个带合并渲染的 Component

需求

  1. 修改数据时触发 render
  2. 同步多次修改只触发一次 render

实现思路

利用 微任务队列 把同一次事件循环里的多次 set 合并到下一帧执行。

js 复制代码
class Component {
  data = { name: '' }
  _pending = false          // 标记位:是否有未 flush 的修改

  constructor() {
    // 通过 Proxy 拦截所有属性写入
    this.data = new Proxy(this.data, {
      set: (target, key, value) => {
        target[key] = value
        this.scheduleRender()
        return true
      }
    })
  }

  scheduleRender() {
    if (this._pending) return   // 已在队列中,跳过
    this._pending = true
    queueMicrotask(() => {
      this.render()
      this._pending = false
    })
  }

  render() {
    console.log(`render - name: ${this.data.name}`)
  }
}

// 测试
const com = new Component()
com.data.name = '张三'
com.data.name = '李四'
com.data.name = '王五'
setTimeout(() => com.data.name = '巷子', 0)

输出顺序:

arduino 复制代码
render - name: 王五      // 第一帧批处理
render - name: 巷子      // setTimeout 宏任务

核心原理:queueMicrotask 把多次写操作合并到同一微任务阶段,只执行一次 render

二、Vue 中同步 vs 异步赋值到底渲染几次?

代码一:同步 for 循环

vue 复制代码
<script setup>
import { ref } from 'vue'
const rCount = ref(0)
for (let i = 1; i <= 5; ++i) {
  rCount.value = i
}
</script>

渲染次数:2 次

  1. 初始挂载:渲染 0
  2. 批处理队列:5 次赋值被合并,最终渲染 5

Vue 内部使用 异步队列(queueJob) 收集同步变更,下一事件循环统一 flush。同一代码块内无论改多少次,都只走一次 DOM diff。

代码二:setTimeout 异步循环

vue 复制代码
<script setup>
import { ref } from 'vue'
const rCount = ref(0)
for (let i = 1; i <= 5; ++i) {
  setTimeout(() => rCount.value = i, 0)
}
</script>

渲染次数:6 次

  1. 初始挂载:渲染 0
  2. 每个 setTimeout 回调都是一个独立宏任务,Vue 的批处理无法跨任务合并,于是 5 次回调触发 5 次独立渲染。

总结

同步代码 → 全部进入同一批处理队列 → 1 次渲染

异步代码 → 每次回调独立任务 → n 次渲染

相关推荐
绝无仅有2 分钟前
后端工程师面试常见问题与回答解析总结
后端·面试·github
程序员爱钓鱼2 小时前
Go语言实战案例 — 项目实战篇:简易博客系统(支持评论)
前端·后端·go
excel9 小时前
ES6 中函数的双重调用方式:fn() 与 fn\...``
前端
可乐爱宅着9 小时前
全栈框架next.js入手指南
前端·next.js
bobz9659 小时前
进程和线程结构体的统一和差异
面试
你的人类朋友11 小时前
什么是API签名?
前端·后端·安全
会豪13 小时前
Electron-Vite (一)快速构建桌面应用
前端
中微子13 小时前
React 执行阶段与渲染机制详解(基于 React 18+ 官方文档)
前端
唐某人丶13 小时前
教你如何用 JS 实现 Agent 系统(2)—— 开发 ReAct 版本的“深度搜索”
前端·人工智能·aigc
中微子13 小时前
深入剖析 useState产生的 setState的完整执行流程
前端