v-memo 是 Vue 3 提供的一个性能优化工具,能帮助开发者缓存模板内容,减少不必要的渲染开销。本文将介绍 v-memo 的引入版本、作用、使用方法和实现原理,并通过示例说明如何使用它。内容基于 Vue 3.5.18(截至 2025 年 7 月的最新版本),各位前端同胞过过眼就好。

一、v-memo 是什么?哪个版本引入?

1. 引入版本

v-memoVue 3.2.0(2021 年 8 月发布)中首次亮相,是 Vue 3 编译器优化的新特性,专为提升渲染性能设计。

2. 作用

v-memo 用于缓存模板的子树,通过传入一个固定长度的依赖数组,只有当数组中的值发生变化时,包裹的 DOM 子树才会重新渲染。其核心优势包括:

  • 减少渲染开销:跳过静态或低频变化内容的重复渲染。
  • 优化复杂场景:特别适合大型列表、表格或包含复杂计算的模板。
  • 灵活控制:相比 v-once 的完全静态化,v-memo 能根据依赖动态决定是否更新。

它类似于 React 的 React.memo,但作用于模板片段,粒度更细,适合微优化。

适用场景

  • 静态 UI(如页头、页脚)。
  • 大型 v-for 列表(例如超过 1000 项)vue官方建议。
  • 包含耗时计算(如 computed 属性)的模板。

二、如何使用 v-memo

1. 基本用法

v-memo 需要一个固定长度的 依赖数组(类型:any[]),只有数组中的值变化时,Vue 才会重新渲染包裹的模板内容。如果依赖不变,渲染将完全跳过,甚至虚拟 DOM 的 VNode 创建也会被省略。

<template><div v-memo="[valueA, valueB]"><!-- 只有 valueA 或 valueB 变时,这部分才会重新渲染 --><p>{{ content }}</p></div>
</template>

注意

  • 正确指定依赖数组至关重要,漏掉关键依赖可能导致更新被错误跳过。
  • 传入空数组(v-memo="[]")等同于 v-once,模板只渲染一次。

2. 示例:优化静态内容

以下示例展示如何用 v-memo 优化包含复杂计算的静态内容。

<template><div><button @click="count++">点我加1:{{ count }}</button><div v-memo="[staticValue]"><p>静态内容:{{ staticValue }}</p><p>复杂计算:{{ expensiveComputed }}</p></div></div>
</template><script setup>
import { ref, computed } from 'vue'const count = ref(0)
const staticValue = ref('固定内容')// 模拟耗时计算
const expensiveComputed = computed(() => {console.log('跑了一次复杂计算')return staticValue.value + ' - 计算结果'
})
</script><style scoped>
button {padding: 8px 16px;background: #409eff;color: white;border: none;border-radius: 4px;cursor: pointer;margin-bottom: 16px;
}
div {padding: 10px;background: #f5f5f5;border-radius: 4px;
}
</style>

运行效果

  • 点击按钮增加 count 会触发父组件渲染,但因为 staticValue 未变,v-memo 包裹的 <div> 及其子内容不会重新渲染。
  • expensiveComputed 仅在初次渲染或 staticValue 变化时运行,控制台只打印一次“跑了一次复杂计算”。

3. 示例:搭配 v-for 优化大型列表

v-memo 在渲染海量 v-for 列表时特别有用。以下示例展示如何优化列表项的渲染。

<template><div><button @click="selected = selected === 1 ? 2 : 1">切换选中:{{ selected }}</button><ul><li v-for="item in list" :key="item.id" v-memo="[item.id === selected]"><p>ID:{{ item.id }} - 选中:{{ item.id === selected }}</p><p>更多内容...</p></li></ul></div>
</template><script setup>
import { ref } from 'vue'const selected = ref(1)
const list = ref([{ id: 1, name: 'Item 1' },{ id: 2, name: 'Item 2' },{ id: 3, name: 'Item 3' },
])
</script><style scoped>
button {padding: 8px 16px;background: #409eff;color: white;border: none;border-radius: 4px;cursor: pointer;margin-bottom: 16px;
}
ul {list-style: none;padding: 0;
}
li {padding: 10px;background: #f5f5f5;border-radius: 4px;margin-bottom: 8px;
}
</style>

说明

  • 每个列表项的 v-memo="[item.id === selected]" 确保只有当 item.id === selected 的值变化时,该项才会重新渲染。
  • item.id 不需加入依赖数组,因为 Vue 会根据 :key 自动处理列表项的复用。
  • selected 变化时,只有受影响的列表项会更新,其他项复用缓存的 VNode,跳过 diff 和渲染。

4. 注意事项

  • 依赖数组准确性:依赖数组必须包含所有影响渲染的变量,否则可能导致更新被错误跳过。
  • v-for 的搭配v-memov-for 需绑定在同一元素上,且不能在 v-for 内部使用 v-memo
    <!-- 正确 -->
    <li v-for="item in list" :key="item.id" v-memo="[item.id === selected]">...</li>
    <!-- 错误 -->
    <ul v-for="item in list" :key="item.id"><li v-memo="[item.id === selected]">...</li>
    </ul>
    
  • 响应式依赖:依赖项需为 refreactivecomputed 值,确保响应式追踪生效。
  • 子组件限制v-memo 只控制模板渲染,子组件内部逻辑可能仍需单独优化。
  • 空依赖数组v-memo="[]" 等同于 v-once,仅渲染一次。

三、v-memo 的实现原理

1. 工作流程

v-memo 结合 Vue 3 的编译器和响应式系统,实现高效的渲染缓存:

  1. 模板解析

    • 编译器识别 v-memo 指令,为包裹的模板生成特殊渲染逻辑。
    • 依赖数组被解析为响应式依赖,类似 watchEffect 的依赖收集。
  2. 依赖追踪

    • 依赖数组中的值由 Vue 3 的 Proxy 响应式系统监控。
    • 依赖变化时,Vue 标记该 DOM 子树需要更新。
  3. VNode 缓存

    • 初次渲染生成 v-memo 包裹子树的 VNode(虚拟 DOM 节点)。
    • 依赖不变时,Vue 复用缓存的 VNode,跳过 diff 和 DOM 更新。
  4. 更新机制

    • 依赖变化时,Vue 生成新 VNode,执行 diff 算法更新 DOM。
    • 子组件的更新逻辑不受 v-memo 控制,需单独优化(如 defineAsyncComponent)。

2. 与 v-once 的区别

  • v-once:只渲染一次,之后完全静态,忽略所有数据变化。
  • v-memo:根据依赖数组动态决定是否渲染,适合半静态场景。

3. 底层细节

  • 响应式系统:依赖 Vue 3 的 Proxy 实现精确的依赖追踪。
  • 编译器优化:为 v-memo 节点添加 PatchFlags,运行时跳过不必要的 diff。
  • 性能提升:通过缓存 VNode 和减少 DOM 操作,显著降低复杂场景的开销。

四、适用场景与优化建议

1. 适用场景

  • 静态 UI:页头、页脚、固定文本。
  • 大型列表:超过 1000 项的 v-for 列表,优化未选中项的渲染。
  • 复杂计算:包裹耗时的 computed 或复杂模板逻辑。

2. 优化建议

  • 精确依赖:只列出影响渲染的变量,避免多余更新。
  • 搭配 v-for:在列表项上使用 v-memo,结合 :key 优化。
  • 其他优化手段
    • defineAsyncComponent 异步加载子组件。
    • 大数据量场景搭配虚拟列表(如 vue-virtual-scroller)。
    • 使用 keep-alive 缓存动态组件。
  • 调试技巧:用控制台日志检查依赖变化,确保 v-memo 行为符合预期。

3. 局限性

  • 子组件更新v-memo 只管模板,子组件逻辑需单独优化。
  • 依赖管理:依赖数组不准确可能导致更新丢失,需仔细设计。
  • 适用范围:频繁变化的场景中,v-memo 的缓存效果有限。

五、总结

  • 版本v-memo 从 Vue 3.2.0 开始支持。
  • 作用:缓存模板子树,仅在依赖变化时更新,优化性能。
  • 用法:用 v-memo="[dep1, dep2]" 包裹模板,指定依赖。
  • 原理:结合 Vue 3 编译器和响应式系统,缓存 VNode,跳过不必要的 diff。

v-memo 是 Vue 3 的性能优化利器,适合处理静态或低频变化的模板,尤其在大型列表或复杂计算场景下效果显著。合理选择依赖,结合其他优化手段,能让项目更高效。

点个收藏,关注前端结城,一起用代码点亮前端世界!🚀

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/web/90744.shtml
繁体地址,请注明出处:http://hk.pswp.cn/web/90744.shtml
英文地址,请注明出处:http://en.pswp.cn/web/90744.shtml

如若内容造成侵权/违法违规/事实不符,请联系英文站点网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

标准库开发和寄存器开发的区别

1.标准库void GPIO_Toggle_INIT(void)//初始化GPIO {GPIO_InitTypeDef GPIO_InitStructure {0};//定义GPIO结构体RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC, ENABLE);//使能GPIO时钟GPIO_InitStructure.GPIO_Pin GPIO_Pin_2;//GPIO引脚选择GPIO_InitStructure.GPIO_Mode …

在 WebSocket 中使用 @Autowired 时遇到空指针异常

背景&#xff1a;在websocket在有新的连接加入进来时&#xff0c;调用servier中的服务&#xff0c;使用 Autowired 注入的 Bean 竟然是 null&#xff01;这并非 Spring 的 Bug&#xff0c;而是对 WebSocket 生命周期管理理解不足导致的。了解这个问题&#xff0c;我们需要区分两…

MGER实验

一、实验拓扑图二、配置1.R5为ISP&#xff0c;只能进行IP地址配置&#xff0c;其所有地址均配为公有IP地址R1侧为15.1.1.1&#xff0c;对应R5为15.1.1.2R2侧为25.1.1.2&#xff0c;对应R5为25.1.1.1R3侧为35.1.1.2&#xff0c;对应R5为35.1.1.1R4侧为45.1.1.2&#xff0c;对应R…

基于 XGBoost 与 SHAP 的医疗自动化办公与可视化系统(下)

— 登录接口 — @app.post(“/token”) def login(form_data: OAuth2PasswordRequestForm = Depends()): user = fake_users_db.get(form_data.username) if not user or form_data.password != user[“password”]: raise HTTPException(status_code=400, detail=“用户名或密…

python学智能算法(二十九)|SVM-拉格朗日函数求解中-KKT条件

引言 前序学习进程中&#xff0c;对拉格朗日函数执行了初步求导&#xff0c;并获得了简化后的拉格朗日函数极值计算式&#xff1a; L(w,b,α)∑i1mαi−12∑i,j1mαiαjyiyjxiTxjL(w,b,\alpha)\sum_{i1}^{m}\alpha_{i}-\frac{1}{2}\sum_{i,j1}^{m}\alpha_{i}\alpha_{j}y_{i}y_…

【AI论文】MiroMind-M1:通过情境感知多阶段策略优化实现数学推理的开源新进展

摘要&#xff1a;近期&#xff0c;大型语言模型已从流畅的文本生成发展至能在多个领域进行高级推理&#xff0c;由此催生了推理语言模型&#xff08;RLMs&#xff09;。在众多领域中&#xff0c;数学推理堪称代表性基准&#xff0c;因为它需要精确的多步骤逻辑与抽象推理能力&a…

《使用Qt Quick从零构建AI螺丝瑕疵检测系统》——6. 传统算法实战:用OpenCV测量螺丝尺寸

目录一、概述1.1 背景介绍&#xff1a;从“看见”到“看懂”1.2 学习目标二、图像预处理&#xff1a;让目标更突出三、轮廓发现与尺寸测量四、总结与展望一、概述 1.1 背景介绍&#xff1a;从“看见”到“看懂” 在上一篇文章中&#xff0c;我们成功地为应用程序安装了“眼睛…

《人性的弱点》重构【01】

手上有本《人性的弱点》&#xff08;韩文桥 译&#xff0c;浙江文艺出版社&#xff0c;2017.1出版&#xff09;&#xff0c;前些年买的&#xff0c;近期翻出来看看。这门书虽成书于80多年前&#xff0c;但卡耐基对人性洞察之深刻&#xff0c;时至今日&#xff0c;并未觉得过时。…

k8s开启审计日志

k8s默认是关闭审计功能的&#xff0c;想看的话需要到apiserver的pod中才可以。 开启此功能是为了进行k8s审计日志的收集&#xff0c;方便我们查看k8s中用户的各自操作。 开启此功能之前&#xff0c;我们要先创建个审计策略文件audit-policy.yaml 例如以下的测验文件 apiVersion…

Kafka MQ 消费者应用场景

Kafka MQ 消费者应用场景 1 消费者自动提交的时机 在 Kafka 中默认的消费位移的提交方式是自动提交,这个由消费者客户端参数 enable.auto.commit 配置,默认值为 true。当然这个默认的自动提交不是每消费一条消息就提交一次,而是定期提交,这个定期的周期时间由客户端参数 …

Git版本控制系统

Git作为目前最流行的分布式版本控制系统&#xff0c;已经成为开发者必备的技能之一。本文将全面介绍Git的核心概念、基本操作、分支管理以及与GitHub的协作开发&#xff0c;帮助读者从零开始掌握Git的使用。 一、Git概述 1.1 Git发展历史 Git诞生于2005年&#xff0c;由Linu…

如何编译RustDesk(Unbuntu 和Android版本)

编译Linux版本的RustDesk备注&#xff1a;官方文档上&#xff0c;一边都是基于sciter&#xff0c;这个在后面已经不建议使用了&#xff0c;但是依然可以编译刚开始的时候看官方的文档&#xff0c;涉及的东西比较多&#xff0c;也搞的一头雾水&#xff0c;通过B站上一个视频&…

Spring中的循环依赖:解密、破局与架构启示

> 当两个Bean紧紧相拥,Spring容器却陷入死锁——这是Java开发者的经典噩梦 某电商平台凌晨上线时突然宕机,日志里反复滚动着`BeanCurrentlyInCreationException`的报错。经排查,**优惠券服务与库存服务在初始化时相互依赖**,形成致命闭环。这个价值百万的故障案例,揭开…

DataFrame​(数据框)

一种二维表格型数据结构&#xff0c;类似于电子表格&#xff08;如 Excel&#xff09;或 SQL 表&#xff0c;由行&#xff08;记录&#xff09;​和列&#xff08;字段&#xff09;​组成。它是数据分析、机器学习和科学计算中最常用的数据结构之一&#xff0c;尤其在 ​Python…

B站视频评论数据爬取

爬取B站视频评论数据爬取与分析 如果只要单纯的脚本可以直接看项目结构里的b_comments.py 一、技术架构 1、环境配置 Python 3.8PyCharm 2、模块配置 requests&#xff1a;用于发送HTTP请求time&#xff1a;用于处理时间相关的操作csv&#xff1a;用于读写CSV文件json&#xff…

OpenAI最新大模型GPT-4o体验之Code Copilot AI编程大模型

一、前言GPT-4o&#xff08;"o"代表"全能"&#xff09;具备处理各种文本、声音和图像资料的能力&#xff0c;能够输出多种格式的文本、声音和图像。GPT-4o 的推出标志着 AI 技术的重大突破。它不再局限于单一媒介&#xff0c;而是首次实现了文本、语音和图…

社交电商推客系统全栈开发指南:SpringCloud+分润算法+Flutter跨端

一、推客系统概述与市场背景推客系统&#xff08;TuiKe System&#xff09;是一种基于社交关系的营销推广平台&#xff0c;通过用户分享商品或服务链接&#xff0c;实现裂变式传播和精准营销。近年来&#xff0c;随着社交电商的蓬勃发展&#xff0c;推客系统已成为企业获客的重…

网安-中间件-Redis未授权访问漏洞

目录 Redis Redis持久化 动态修改配置 使用反弹连接的情况 常见监听端口的方式 常见建立反弹连接的方式 流程 Linux crontab cron文件存储路径 利用Redis实现攻击 1.webshell提权案例 2.定时任务shell反弹案例 3.SSH Key getshell案例 ​编辑Redis其他利用方式 …

【c++深入系列】:万字详解栈和队列和deque(附模拟实现的源码)

&#x1f525; 本文专栏&#xff1a;c &#x1f338;作者主页&#xff1a;努力努力再努力wz &#x1f4aa; 今日博客励志语录&#xff1a; 石头能被水滴穿&#xff0c;不是因为水有多强&#xff0c;而是因为它从未停过。 ★★★ 本文前置知识&#xff1a; 模版 栈 那么栈这个…

速通python加密之RSA加密

RSA加密 RSA加密是一种非对称加密算法&#xff08;与AES等对称加密不同&#xff09;&#xff0c;由罗纳德李维斯特&#xff08;Ron Rivest&#xff09;、阿迪萨莫尔&#xff08;Adi Shamir&#xff09;和伦纳德阿德曼&#xff08;Leonard Adleman&#xff09;于1977年提出&…