Vue 3 Diff 算法受 `v-for` 循环中的 `key` 属性影响
Vue 3 的 Diff 算法会受到 v-for 循环中的 key 属性的影响,key 的选择直接关系到 Diff 算法的效率和最终的 DOM 更新结果。
key 的作用
在 Vue 中,key 是一种标识,它用于唯一标记每个虚拟 DOM 节点。Diff 算法会根据 key 判断新旧节点是否是同一个节点。
没有 key 时
- 如果没有
key,Vue 会默认使用节点的索引值作为标识。 - 当列表发生变化时,由于索引可能对应了错误的节点,会导致无法正确复用现有节点,从而增加不必要的 DOM 操作。
有 key 时
- 当指定了唯一的
key,Vue 可以准确匹配新旧节点。 - 这样可以复用相同
key的节点,避免误匹配或冗余操作,提高性能。
Diff 算法如何处理 key
-
key匹配规则- 在 Diff 过程中,如果新旧节点的
key相同,Vue 会认为它们是相同的节点,执行更新操作而非重建。 - 如果
key不同,Vue 会移除旧节点并创建新节点。
- 在 Diff 过程中,如果新旧节点的
-
有
key的场景- Vue 通过
key快速定位新旧节点在列表中的位置,避免盲目比较,从而优化性能。 - Vue 3 使用 最长递增子序列(LIS) 算法来减少 DOM 操作,
key的稳定性是实现该优化的重要前提。
- Vue 通过
-
没有
key的场景- Vue 会退化为按索引逐一比较的模式。
- 如果数据顺序发生变化,可能会导致大量的节点重建和移动。
案例分析
示例 1:没有 key 的情况
<template><div><div v-for="item in items">{{ item }}</div></div>
</template><script>
export default {data() {return {items: [1, 2, 3]};},mounted() {setTimeout(() => {this.items = [3, 2, 1]; // 改变顺序}, 1000);}
};
</script>
结果:
- Vue 会将
[1, 2, 3]和[3, 2, 1]按索引逐个对比。 - 由于没有
key,节点内容不同,导致所有节点被替换,性能较差。
示例 2:使用唯一的 key
<template><div><div v-for="item in items" :key="item">{{ item }}</div></div>
</template><script>
export default {data() {return {items: [1, 2, 3]};},mounted() {setTimeout(() => {this.items = [3, 2, 1]; // 改变顺序}, 1000);}
};
</script>
结果:
- Vue 根据
key匹配节点[1, 2, 3]和[3, 2, 1],只移动节点位置,而不会销毁重建。 - 这样可以显著减少 DOM 操作,提高性能。
key 影响 Diff 的性能与结果
-
性能影响
- 没有
key:Vue 需要逐个比对每个节点的内容,无法充分复用节点。 - 有
key:Vue 可以快速确定哪些节点需要更新、移动或删除,从而减少不必要的 DOM 操作。
- 没有
-
更新结果的准确性
- 没有
key:可能会导致节点错误复用,渲染结果不符合预期。 - 有
key:可以确保每个节点的复用和更新都是正确的。
- 没有
<template><div class="ForKeyInDiffVNode"><el-button @click="change">改变值</el-button><div class="item-class" v-for="(item, idx) in lists" :key="idx"><el-checkbox></el-checkbox><span>{{ item.name }}</span></div></div>
</template><script setup lang="ts">
import {ref} from "vue";const lists = ref([{name: 'hmk',age: 20,show: false},{name: '张三',age: 21,show: false},{name: '李四',age: 22,show: false},{name: '王五',age: 23,show: false}
])
// 之前选中第三个,点击删除第三个之前的数据后,会出现错乱情况
// 给key加上一个唯一的id即可解决问题
const change = () => {lists.value.splice(1, 1)
}</script>
最佳实践:key 的选择
-
使用唯一标识符
- 使用数据中的唯一标识(如 ID)作为
key:<div v-for="item in items" :key="item.id">{{ item.name }}</div> - 避免使用非唯一的值(如索引)作为
key,因为数据顺序变化时可能会导致错误复用。
- 使用数据中的唯一标识(如 ID)作为
-
避免使用索引作为
key- 索引是动态的,当列表顺序改变或元素插入时,索引会变化,导致 Diff 结果不准确:
<div v-for="item in items" :key="index">{{ item }}</div> - 如果数据内容变化频繁,索引
key会导致不必要的 DOM 重建。
- 索引是动态的,当列表顺序改变或元素插入时,索引会变化,导致 Diff 结果不准确:
-
保证
key的稳定性key应该在组件的整个生命周期内保持不变,否则会导致 Diff 结果错误。
总结
-
key对 Diff 算法至关重要:- 没有
key:Vue 会按索引匹配,导致错误复用和性能下降。 - 有
key:Vue 可高效、准确地更新节点,减少 DOM 操作。
- 没有
-
选择唯一且稳定的
key:- 推荐使用数据中的唯一标识符(如
id)。 - 避免使用索引或动态值作为
key。
- 推荐使用数据中的唯一标识符(如
-
Diff 性能优化:
- Vue 3 使用
key时可以充分利用最长递增子序列(LIS)优化算法,最小化 DOM 移动和更新操作,提高性能。
- Vue 3 使用
相关文章:
Vue 3 Diff 算法受 `v-for` 循环中的 `key` 属性影响
Vue 3 的 Diff 算法会受到 v-for 循环中的 key 属性的影响,key 的选择直接关系到 Diff 算法的效率和最终的 DOM 更新结果。 key 的作用 在 Vue 中,key 是一种标识,它用于唯一标记每个虚拟 DOM 节点。Diff 算法会根据 key 判断新旧节点是否是…...
江科大STM32入门——看门狗笔记整理
wx:嵌入式工程师成长日记 (一)简介 WDG(Watchdog)看门狗看门狗可以监控程序的运行状态,当程序因为设计漏洞(无法预料)、硬件故障、电磁干扰等原因,出现卡死或跑飞现象时,看门狗能及…...
【计算机网络】lab7 TCP协议
🌈 个人主页:十二月的猫-CSDN博客 🔥 系列专栏: 🏀计算机网络_十二月的猫的博客-CSDN博客 💪🏻 十二月的寒冬阻挡不了春天的脚步,十二点的黑夜遮蔽不住黎明的曙光 目录 1. 实验目的…...
计算机视觉:解锁未来智能世界的钥匙
计算机视觉:解锁未来智能世界的钥匙 在信息技术飞速发展的今天,计算机视觉作为人工智能领域的一个重要分支,正以前所未有的速度改变着我们的生活与工作方式。它使机器能够“看”并理解图像和视频中的信息,为自动驾驶、医疗影像分…...
Java的Stream流和Option类
1. Stream 流 背景 Stream是Java 8引入的一个用于处理集合(或其他数据源)中的元素的API。它提供了一种声明式的方式来处理数据,并可以链式调用。Stream支持惰性求值,也支持并行流处理。 1.1 创建 Stream 创建一个Stream可以通…...
深入理解ASP.NET Core 管道的工作原理
在 .NET Core 中,管道(Pipeline)是处理 HTTP 请求和响应的中间件组件的有序集合。每个中间件组件都可以对请求进行处理,并将其传递给下一个中间件组件,直到请求到达最终的处理程序。管道的概念类似于流水线,…...
多模态论文笔记——CLIP
大家好,这里是好评笔记,公主号:Goodnote,专栏文章私信限时Free。本文详细介绍这几年AIGC火爆的隐藏功臣,多模态模型:CLIP。 文章目录 CLIP(Contrastive Language-Image Pre-training)…...
brpc之baidu_protocol
简介 是brpc默认使用的协议 初始化 Protocol baidu_protocol { ParseRpcMessage,SerializeRequestDefault, PackRpcRequest,ProcessRpcRequest, ProcessRpcResponse,VerifyRpcRequest, NULL, NULL,CONNECTION_TYPE_ALL, "baidu_std" };协议定义 定义在baidu_rpc…...
LeetCode:39. 组合总和
跟着carl学算法,本系列博客仅做个人记录,建议大家都去看carl本人的博客,写的真的很好的! 代码随想录 LeetCode:39. 组合总和 给你一个 无重复元素 的整数数组 candidates 和一个目标整数 target ,找出 cand…...
SOLID原则学习,开闭原则(Open Closed Principle, OCP)
文章目录 1. 定义2. 开闭原则的详细解释3. 实现开闭原则的方法4. 总结 1. 定义 开闭原则(Open-Closed Principle,OCP)是面向对象设计中的五大原则(SOLID)之一,由Bertrand Meyer提出。开闭原则的核心思想是…...
Unreal Engine 5 C++ Advanced Action RPG 七章笔记
第七章 Ranged Enemy 2-Ranged Enemy Starting Weapon 制作新敌人的流程准备 新敌人的武器起始的状态数据自己的战斗能力投射能力自己的行为树 创建角色,添加武器,添加数据,就是继承之前的基类敌人的 运行结果 3-Glacer Starting Stats 看看就行,就是复制曲线表格更改数…...
自动连接校园网wifi脚本实践(自动网页认证)
目录 起因执行步骤分析校园网登录逻辑如何判断当前是否处于未登录状态? 书写代码打包设置开机自动启动 起因 我们一般通过远程控制的方式访问实验室电脑,但是最近实验室老是断电,但重启后也不会自动连接校园网账户认证,远程工具&…...
HTTP/HTTPS ⑤-CA证书 || 中间人攻击 || SSL/TLS
这里是Themberfue ✨上节课我们聊到了对称加密和非对称加密,实际上,单纯地非对称加密并不能保证数据不被窃取,我们还需要一个更加重要的东西——证书 中间人攻击 通过非对称加密生成私钥priKey和公钥pubKey用来加密对称加密生成的密钥&…...
traceroute原理探究
文章中有截图,看不清的话,可以把浏览器显示比例放大到200%后观看。 linux下traceroute的原理 本文通过抓包观察一下linux下traceroute的原理 环境:一台嵌入式linux设备,内网ip是192.168.186.195,其上有192.168.202.…...
50_Lua垃圾回收
1.Lua垃圾回收机制概述 Lua采用了一种自动内存管理机制,称为垃圾回收(Garbage Collection, GC)。垃圾回收的主要目的是回收程序中不再被使用的内存,从而避免内存泄漏。Lua的垃圾回收器负责回收动态分配的对象,如函数、用户数据、表、字符串、线程、内部结构等。Lua的垃圾…...
Git-2-:Cherry-Pick 的使用场景及使用流程
前面我们说了 Git合并、解决冲突、强行回退等解决方案 >> 点击查看 这里再说一下 Cherry-Pick功能,Cherry-Pick不是merge,只是把部分功能代码Cherry-Pick到远程的目标分支 git cherry-pick功能简介: git cherry-pick 是用来从一个分…...
【C++】21.map和set的使用
文章目录 1. 序列式容器和关联式容器2. set系列的使用2.1 set和multiset参考文档2.2 set类的介绍2.3 set的构造和迭代器构造函数:双向迭代器迭代器: 2.4 set的增删查2.5 insert和迭代器遍历使用样例:2.6 find和erase使用样例:2.7 …...
burpsiute的基础使用(2)
爆破模块(intruder): csrf请求伪造访问(模拟攻击): 方法一: 通过burp将修改,删除等行为的数据包压缩成一个可访问链接,通过本地浏览器访问(该浏览器用户处于登陆状态&a…...
ElasticSearch 同义词匹配
synonym.txt 电脑, 计算机, 主机 复印纸, 打印纸, A4纸, 纸, A3 平板电脑, Pad DELETE /es_sku_index_20_20250109 PUT /es_sku_index_20_20250109 {"settings": {"index": {"number_of_shards": "5","number_of_replicas&quo…...
linux RT-Preempt spin lock实现
一、spin_lock概述 Spinlock是linux内核中常用的一种互斥锁机制,和mutex不同,当无法持锁进入临界区的时候,当前执行线索不会阻塞,而是不断的自旋等待该锁释放。正因为如此,自旋锁也是可以用在中断上下文的。也正是因为…...
YSYX学习记录(八)
C语言,练习0: 先创建一个文件夹,我用的是物理机: 安装build-essential 练习1: 我注释掉了 #include <stdio.h> 出现下面错误 在你的文本编辑器中打开ex1文件,随机修改或删除一部分,之后…...
MVC 数据库
MVC 数据库 引言 在软件开发领域,Model-View-Controller(MVC)是一种流行的软件架构模式,它将应用程序分为三个核心组件:模型(Model)、视图(View)和控制器(Controller)。这种模式有助于提高代码的可维护性和可扩展性。本文将深入探讨MVC架构与数据库之间的关系,以…...
CMake 从 GitHub 下载第三方库并使用
有时我们希望直接使用 GitHub 上的开源库,而不想手动下载、编译和安装。 可以利用 CMake 提供的 FetchContent 模块来实现自动下载、构建和链接第三方库。 FetchContent 命令官方文档✅ 示例代码 我们将以 fmt 这个流行的格式化库为例,演示如何: 使用 FetchContent 从 GitH…...
浅谈不同二分算法的查找情况
二分算法原理比较简单,但是实际的算法模板却有很多,这一切都源于二分查找问题中的复杂情况和二分算法的边界处理,以下是博主对一些二分算法查找的情况分析。 需要说明的是,以下二分算法都是基于有序序列为升序有序的情况…...
css3笔记 (1) 自用
outline: none 用于移除元素获得焦点时默认的轮廓线 broder:0 用于移除边框 font-size:0 用于设置字体不显示 list-style: none 消除<li> 标签默认样式 margin: xx auto 版心居中 width:100% 通栏 vertical-align 作用于行内元素 / 表格单元格ÿ…...
智能AI电话机器人系统的识别能力现状与发展水平
一、引言 随着人工智能技术的飞速发展,AI电话机器人系统已经从简单的自动应答工具演变为具备复杂交互能力的智能助手。这类系统结合了语音识别、自然语言处理、情感计算和机器学习等多项前沿技术,在客户服务、营销推广、信息查询等领域发挥着越来越重要…...
FFmpeg:Windows系统小白安装及其使用
一、安装 1.访问官网 Download FFmpeg 2.点击版本目录 3.选择版本点击安装 注意这里选择的是【release buids】,注意左上角标题 例如我安装在目录 F:\FFmpeg 4.解压 5.添加环境变量 把你解压后的bin目录(即exe所在文件夹)加入系统变量…...
Ubuntu Cursor升级成v1.0
0. 当前版本低 使用当前 Cursor v0.50时 GitHub Copilot Chat 打不开,快捷键也不好用,当看到 Cursor 升级后,还是蛮高兴的 1. 下载 Cursor 下载地址:https://www.cursor.com/cn/downloads 点击下载 Linux (x64) ,…...
华为OD机试-最短木板长度-二分法(A卷,100分)
此题是一个最大化最小值的典型例题, 因为搜索范围是有界的,上界最大木板长度补充的全部木料长度,下界最小木板长度; 即left0,right10^6; 我们可以设置一个候选值x(mid),将木板的长度全部都补充到x,如果成功…...
软件工程 期末复习
瀑布模型:计划 螺旋模型:风险低 原型模型: 用户反馈 喷泉模型:代码复用 高内聚 低耦合:模块内部功能紧密 模块之间依赖程度小 高内聚:指的是一个模块内部的功能应该紧密相关。换句话说,一个模块应当只实现单一的功能…...
