vue中使用v-for时为什么不能用index作为key


Posted in Javascript onApril 04, 2020

结论:

  • 更新DOM的时候会出现性能问题
  • 会发生一些状态bug
  • React 中的 key 也是如此
  • 如果已经了解 为什么要用key,可以通过目录直接跳到下一节。

为什么要用key?

Vue 和 React 都实现了一套虚拟DOM,使我们可以不直接操作DOM元素,只操作数据便可以重新渲染页面。而隐藏在背后的原理便是其高效的Diff算法。

Vue 和 React 的虚拟DOM的Diff算法大致相同,其核心是基于两个简单的假设:

  1. 两个相同的组件产生类似的DOM结构,不同的组件产生不同的DOM结构。
  2. 同一层级的一组节点,他们可以通过唯一的id进行区分。

基于以上这两点假设,使得虚拟DOM的Diff算法的复杂度从O(n^3)降到了O(n)。

用一张图简单说明一下:

vue中使用v-for时为什么不能用index作为key

当页面的数据发生变化时,Diff算法只会比较同一层级的节点:

如果节点类型不同,直接干掉前面的节点,再创建并插入新的节点。

如果节点类型相同,则会重新设置该节点的属性,从而实现节点的更新。

举个栗子:

vue中使用v-for时为什么不能用index作为key

我们希望可以在B和C之间加一个F,Diff算法默认执行起来是这样的:

vue中使用v-for时为什么不能用index作为key

即把C更新成F,D更新成C,E更新成D,最后再插入E,这样效率不高,且性能不够好。

但是,如果使用key来给每个节点做一个唯一标识,Diff算法就可以正确的识别此节点,找到正确的位置区插入新的节点。

vue中使用v-for时为什么不能用index作为key

总而言之,key的作用主要是为了高效的更新虚拟DOM 。另外vue中在使用相同标签名元素的过渡切换时,也会使用到key属性,其目的也是为了让vue可以区分它们,否则vue只会替换其内部属性而不会触发过渡效果。

这里,也建议尽可能在使用 v-for 时提供 key attribute,除非遍历输出的 DOM 内容非常简单。

为什么不能用index作为key?

举个栗子:

<template>
  <div v-for="(item, index) in list" :key="index" >{{item.name}}</div>
</template>
const list = [
  {
    id: 1,
    name: "Person1"
  },
  {
    id: 2,
    name: "Person2"
  },
  {
    id: 3,
    name: "Person3"
  },
  {
    id:4,
    name:"Person4"
  }
];

此时,删除 “Person4” 是正常的,但是如果我删除 “Person2” 就会出现问题。

删除前

key id index name
0 1 0 Person1
1 2 1 Person2
2 3 2 Person3
3 4 3 Person4

删除后

key id index name
0 1 0 Person1
1 3 1 Person3
2 4 2 Person4

这个时候,除了 Person1 之外,剩下的 Person3、Person4,因为被发现与相应 key 的绑定关系有变化,所以被重新渲染,这会影响性能。
如果此时 list 的 item 是 select 的选项,其中 Person3 是选中的,这个时候 Person2 被删除了,用 index 作为 key 就会变成是 Person4 选中的了,这就产生了bug。

如果使用唯一id作为key,删除 Person2 后,剩下的元素因为与 key 的关系没有发生变化,都不会被重新渲染,从而达到提升性能的目的。此时,list 的 item 作为 select 的选项,也不会出现上面所描述的bug。

到此这篇关于vue中使用v-for时为什么不能用index作为key的文章就介绍到这了,更多相关vue v-for不能用index作为key内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

Javascript 相关文章推荐
jquery删除ID为sNews的tr元素的内容
Apr 10 Javascript
javascript 处理null及null值示例
Jun 09 Javascript
实例分析javascript中的call()和apply()方法
Nov 28 Javascript
javascript日期计算实例分析
Jun 29 Javascript
微信小程序实现图片自适应(支持多图)
Jan 25 Javascript
Vue.js使用$.ajax和vue-resource实现OAuth的注册、登录、注销和API调用
May 10 Javascript
angular-ngSanitize模块-$sanitize服务详解
Jun 13 Javascript
vue cli构建的项目中请求代理与项目打包问题
Feb 26 Javascript
vue1.0和vue2.0的watch监听事件写法详解
Sep 11 Javascript
Javascript摸拟自由落体与上抛运动原理与实现方法详解
Apr 08 Javascript
React服务端渲染原理解析与实践
Mar 04 Javascript
Python机器学习之决策树和随机森林
Jul 15 Javascript
详解为什么Vue中不要用index作为key(diff算法)
Apr 04 #Javascript
原理深度解析Vue的响应式更新比React快
Apr 04 #Javascript
Vue的data、computed、watch源码浅谈
Apr 04 #Javascript
VUE table表格动态添加一列数据,新增的这些数据不可以编辑(v-model绑定的数据不能实时更新)
Apr 03 #Javascript
mpvue实现微信小程序快递单号查询代码
Apr 03 #Javascript
mpvue网易云短信接口实现小程序短信登录的示例代码
Apr 03 #Javascript
javascript用defineProperty实现简单的双向绑定方法
Apr 03 #Javascript
You might like
PHP设计模式之代理模式的深入解析
2013/06/13 PHP
PHP接入微信H5支付的方法示例
2019/10/28 PHP
JavaScript 异步调用框架 (Part 5 - 链式实现)
2009/08/04 Javascript
JavaScript学习笔记记录我的旅程
2012/05/23 Javascript
滚动图片效果 jquery实现回旋滚动效果
2013/01/08 Javascript
关于innerHTML后丢失动态绑定的EVENT问题解决方法
2013/05/19 Javascript
如何让浏览器支持jquery ajax load 前进、后退功能
2014/06/12 Javascript
JQuery做的一个简单的点灯游戏分享
2014/07/16 Javascript
JS模式之简单的订阅者和发布者模式完整实例
2015/06/30 Javascript
jQuery实现简单的点赞效果
2020/05/29 Javascript
微信小程序 使用picker封装省市区三级联动实例代码
2016/10/28 Javascript
纯javascript版日历控件
2016/11/24 Javascript
jQuery常见的选择器及用法介绍
2016/12/20 Javascript
javascript 操作cookies详解及实例
2017/02/22 Javascript
Bootstrap表格制作代码
2017/03/17 Javascript
jQuery事件_动力节点Java学院整理
2017/07/05 jQuery
js+SVG实现动态时钟效果
2018/07/14 Javascript
javascript中一些奇葩的日期换算方法总结
2018/11/14 Javascript
Vuex实现数据增加和删除功能
2019/11/11 Javascript
Antd中单个DatePicker限定时间输入范围操作
2020/10/29 Javascript
VueCli生产环境打包部署跨域失败的解决
2020/11/13 Javascript
[01:20:38]完美世界DOTA2联赛 GXR vs IO 第一场 11.07
2020/11/09 DOTA
python之PyMongo使用总结
2017/05/26 Python
python腾讯语音合成实现过程解析
2019/08/01 Python
python list数据等间隔抽取并新建list存储的例子
2019/11/27 Python
Django restframework 框架认证、权限、限流用法示例
2019/12/21 Python
Python爬虫库BeautifulSoup获取对象(标签)名,属性,内容,注释
2020/01/25 Python
PyCharm最新激活码(2020/10/27全网最新)
2020/10/27 Python
基于Python中Remove函数的用法讨论
2020/12/11 Python
欧洲最大的婴幼儿服装及内衣公司:Petit Bateau(小帆船)
2016/08/16 全球购物
都柏林通行卡/城市通票:The Dublin Pass
2020/02/16 全球购物
建筑工程专业毕业生自荐信
2013/10/19 职场文书
银行竞聘演讲稿
2014/05/16 职场文书
教师岗位职责范本
2015/04/02 职场文书
「Manga Time Kirara MAX」2022年5月号封面公开
2022/03/21 日漫
Python中tqdm的使用和例子
2022/09/23 Python