v-memo是Vue 3.2+专为长列表优化的指令,需配合v-for和稳定key使用,依赖字段级浅比较,禁用对象引用和条件指令包裹,可显著减少patch开销。
v-memo 是 Vue 3.2+ 引入的实验性指令,专为长列表(尤其是 v-for)设计,它不阻止更新,而是让 Vue 在依赖未变时跳过整块子树的 diff 和 patch,从而显著降低虚拟 DOM 计算开销。用对了,性能提升明显;用错了,可能引发错位、状态不同步等隐蔽问题。
v-memo 不能单独存在,必须嵌套在 v-for 项内部,且该项必须有明确、唯一、稳定的 :key(如 item.id)。key 不仅用于 DOM 复用,也与 v-memo 的缓存边界强绑定:
<div v-for="item in list" :key="item.id" v-memo="[item.id, item.status]">
<div v-for="item in list" :key="index" v-memo="[item.id]">(索引不是稳定标识)<div v-for="item in list" :key="item.id" v-memo="[list]">(依赖父级整个数组,一动全动)v-memo 对依赖项做浅层严格相等(===)比对。若传入整个响应式对象(如 item),每次渲染都会生成新引用,导致缓存始终失效:
v-memo="[item.id, item.name, item.isRead]"
v-memo="[computedAvatarUrl]")v-memo="[item]" 或 v-memo="[item.tags]"(tags 是数组或对象,引用易变)null、undefined、NaN 会导致 v-memo 退化为“永不缓存”v-memo 依赖编译时确定的渲染结构。若包裹在条件指令中,Vue 无法保证缓存上下文的一致性,可能导致子树复用错乱:
立即学习“前端免费学习笔记(深入)”;
<div v-if="show" v-memo="[item.id]">...</div>
v-show(保持 DOM 存在)<div v-show="show" v-memo="[item.id]">...</div>
开启 Vue Devtools 的 Performance 面板,触发一次局部更新(如点击某条消息标记已读),观察 patch 节点数:
const { id } = item 后传 id,会断开响应追踪)它不是银弹,但对消息流、商品墙、日志面板这类“结构固定、局部更新”的长列表,v-memo 是目前 Vue 生态中最轻量、最直接的渲染减负手段。