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 相关文章推荐
表单(FORM)的一些实用效果代码
Mar 25 Javascript
javascript AutoScroller 函数类
May 29 Javascript
JS实现点击文字对应DIV层不停闪动效果的方法
Mar 02 Javascript
javascript发送短信验证码实现代码
Nov 12 Javascript
jQuery实现点击弹出背景变暗遮罩效果实例代码
Jun 24 Javascript
JavaScript获取select中text值的方法
Feb 13 Javascript
Vue.js -- 过滤器使用总结
Feb 18 Javascript
Angular 2 ngForm中的ngModel、[ngModel]和[(ngModel)]的写法
Jun 29 Javascript
Angularjs过滤器实现动态搜索与排序功能示例
Dec 13 Javascript
JavaScript获取页面元素的常用方法详解
Sep 28 Javascript
vue el-table实现自定义表头
Dec 11 Javascript
基于Vant UI框架实现时间段选择器
Dec 24 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中大于2038年时间戳的问题处理方案
2015/03/03 PHP
PHP下SSL加密解密、验证、签名方法(很简单)
2020/06/28 PHP
CI框架(ajax分页,全选,反选,不选,批量删除)完整代码详解
2016/11/01 PHP
Laravel学习教程之model validation的使用示例
2017/10/23 PHP
Javascript 阻止javascript事件冒泡,获取控件ID值
2009/06/27 Javascript
在chrome中window.onload事件的一些问题
2010/03/01 Javascript
js setTimeout()函数介绍及应用以倒计时为例
2013/12/12 Javascript
实用框架(iframe)操作代码
2014/10/23 Javascript
nodejs教程之制作一个简单的文章发布系统
2014/11/21 NodeJs
实现音乐播放器的代码(html5+css3+jquery)
2015/08/04 Javascript
JavaScript常用基础知识强化学习
2015/12/09 Javascript
JS提交form表单实例分析
2015/12/10 Javascript
JavaScript实现解析INI文件内容的方法
2016/11/17 Javascript
vue项目中使用lib-flexible解决移动端适配的问题解决
2018/08/23 Javascript
JS二级菜单不同实现方法分析【4种方法】
2018/12/21 Javascript
Webpack设置环境变量的一些误区详解
2019/12/19 Javascript
微信小程序自定义模态弹窗组件详解
2019/12/24 Javascript
[02:33]DOTA2亚洲邀请赛趣味视频之吐真话筒
2018/03/31 DOTA
[42:56]VGJ.S vs Serenity 2018国际邀请赛小组赛BO2 第二场 8.19
2018/08/21 DOTA
利用python获得时间的实例说明
2013/03/25 Python
Python常用模块用法分析
2014/09/08 Python
在Python下进行UDP网络编程的教程
2015/04/29 Python
python对html代码进行escape编码的方法
2015/05/04 Python
Mac中Python 3环境下安装scrapy的方法教程
2017/10/26 Python
Python实现App自动签到领取积分功能
2018/09/29 Python
爬虫代理池Python3WebSpider源代码测试过程解析
2019/12/20 Python
Python通过两个dataframe用for循环求笛卡尔积
2020/04/29 Python
Python使用os.listdir和os.walk获取文件路径
2020/05/21 Python
Python 没有main函数的原因
2020/07/10 Python
俄罗斯最大的在线珠宝大卖场:Nebo
2019/12/08 全球购物
租房协议书
2014/04/10 职场文书
个人对照检查材料思想汇报(四风问题)
2014/09/25 职场文书
秦始皇兵马俑导游词
2015/02/02 职场文书
自主招生专家推荐信
2015/03/26 职场文书
刚学完怎么用Python实现定时任务,转头就跑去撩妹!
2021/06/05 Python
探讨Java中的深浅拷贝问题
2021/06/26 Java/Android