详解如何理解vue的key属性


Posted in Javascript onApril 14, 2019

如果没有这个属性的时候vue应用 in-place patch(就地复用)策略。列表里的顺序发生改变的时候比如shuffle(列表打乱)的时候,vue为了提升性能,不会移动dom元素,只是更新相应元素的内容节点。

就地复用的弊端

这个默认的模式是高效的,但是只适用于不依赖子组件状态或临时 DOM 状态 (例如:表单输入值) 的列表渲染输出。

如上引用自官网,这个模式就是上面的“就地复用”策略。那么是不是依赖子组件状态的列表渲染采用上面的模式就出问题了呢。如下测试代码:

<ul>
  <li v-for="item in items">
   <p>{{ item.title }}</p>
   <p>{{ item.des }}</p>
   <tweet-component></tweet-component>
  </li>
  </ul>
 <script>
  Vue.component("tweet-component", {
  template: `
   <div class="tweet">
   <div class="box">
    {{xixi}}
   </div>
   </div>
  `,
  data() {
   return {
   xixi: Math.random()
   };
  }
  });
  new Vue({
  el: "#app",
  data: {
   items: [
   { title: "nihao1", des: "xiexie1" },
   { title: "nihao2", des: "xiexie2" },
   { title: "nihao3", des: "xiexie3" }
   ]
  },
  methods: {
   shuffle() {
    // lodash的shuffle方法
   this.items = _.shuffle(this.items);
   }
  }
  });
 </script>

操作如下:

详解如何理解vue的key属性 

问题出现了:发生变化时,子组件 没有更新

ps: 测试临时 DOM 状态 (例如:表单输入值)可以参考这个链接 List Rendering and Vue's v-for Directive

key的作用

这个时候引入 key 就可以解决这个问题。 key 的作用是给予一个节点唯一的身份识别,有相同父元素的子元素必须有独特的 key 。这样它可以前后对比,算出哪些节点是要重复使用或者调整顺序。比如原先的 key 的顺序是 i1,i2,i3 ,之后变成了 i2,i1,i3 这个时候只要i3保持不变,把i2 insertBefore 到i1节点前就行了(以上是举例,vue具体怎么操作的需要去研究源码)。如果是利用数组的 index 来作为 key 则两次对比没有区别,就会出现上面动图里出现的子组件没有更新的情况。

可以通过查看下面两个动图查看dom节点的变化。第一个gif是没有使用 key ,第二个是使用了 key 。可以看到我点了按钮后,第一个只是更新 <p> 标签的内容节点。第二个是移动了某一项的 <li> 标签。

详解如何理解vue的key属性详解如何理解vue的key属性

ps: 另外可以通过chrome的dom断点功能查看子元素的插入删除。

DOM 更改断点

key的其他用途

key不是只能用在v-for上还可以用在其他元素上。如下代码

<div v-if="toggle">Hello</div>
<div v-else>Goodbye</div>

切换toggle时,它也是切换div。如果你想用上enter/leave animations时,即节点插入和删除的时候的动画时就要打破这个复用,如下:

<transition>
 <div v-if="toggle" key="1">Hello</div>
 <div v-else key="2">Goodbye</div>
</transition>

总结

如果你的列表不发生变化,或者你只是往列表最后一项添加元素则看似不必要增加 key 字段。但是你的项目中有很多列表的时候,你有的需要加 key ,有的不需要加。还不如所有的都加上呢避免别人理解困难(^._.^)ノ。

参考链接

https://vuejs.org/v2/guide/list.html#key

vuejs.org/v2/api/#key

https://forum.vuejs.org/t/simple-clarification-of-when-using-key-with-v-for-is-appropriate-and-why/28966/5

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
jquery lazyload延迟加载技术的实现原理分析
Jan 24 Javascript
jQuery 图片切换插件(代码比较少)
May 07 Javascript
javascript学习笔记(六) Date 日期类型
Jun 19 Javascript
js格式化货币数据实现代码
Sep 04 Javascript
ECMAScript6新增值比较函数Object.is
Jun 12 Javascript
AngularJS基础 ng-csp 指令详解
Aug 01 Javascript
AngularJS 单元测试(二)详解
Sep 21 Javascript
JS实现简单的天数计算器完整实例
Apr 28 Javascript
基于webpack4.X从零搭建React脚手架的方法步骤
Dec 23 Javascript
Vue 实现简易多行滚动&quot;弹幕&quot;效果
Jan 02 Javascript
Vue生命周期activated之返回上一页不重新请求数据操作
Jul 26 Javascript
antd design table更改某行数据的样式操作
Oct 31 Javascript
axios+Vue实现上传文件显示进度功能
Apr 14 #Javascript
Vue 使用formData方式向后台发送数据的实现
Apr 14 #Javascript
说说如何使用Vuex进行状态管理(小结)
Apr 14 #Javascript
基于Vue2-Calendar改进的日历组件(含中文使用说明)
Apr 14 #Javascript
浅谈Vue页面级缓存解决方案feb-alive(上)
Apr 14 #Javascript
vue 实现小程序或商品秒杀倒计时
Apr 14 #Javascript
js中async函数结合promise的小案例浅析
Apr 14 #Javascript
You might like
生成php程序的php代码
2008/04/07 PHP
理解php Hash函数,增强密码安全
2011/02/25 PHP
ThinkPHP基本的增删查改操作实例教程
2014/08/22 PHP
PHP的拦截器实例分析
2014/11/03 PHP
php一维二维数组键排序方法实例总结
2014/11/13 PHP
php5.2的curl-bug 服务器被php进程卡死问题排查
2016/09/19 PHP
PHP实现基于图的深度优先遍历输出1,2,3...n的全排列功能
2017/11/10 PHP
Jquery:ajax实现翻页无刷新功能代码
2013/08/05 Javascript
Function.prototype.bind用法示例
2013/09/16 Javascript
标题过长使用javascript按字节截取字符串
2014/04/24 Javascript
JQuery获取表格数据示例代码
2014/05/26 Javascript
javascript动态修改Li节点值的方法
2015/01/20 Javascript
使用Script元素发送JSONP请求的方法
2016/06/12 Javascript
Angularjs中controller的三种写法分享
2016/09/21 Javascript
Javascript之深入浅出prototype
2017/02/06 Javascript
基于Bootstrap模态对话框只加载一次 remote 数据的解决方法
2017/07/09 Javascript
详解最新vue-cli 2.9.1的webpack存在问题
2017/12/16 Javascript
详解vue mixins和extends的巧妙用法
2017/12/20 Javascript
vue-lazyload图片延迟加载插件的实例讲解
2018/02/09 Javascript
vue 1.0 结合animate.css定义动画效果
2018/07/11 Javascript
原生JS实现$.param() 函数的方法
2018/08/10 Javascript
Vue常用指令详解分析
2018/08/19 Javascript
vue+axios 前端实现登录拦截的两种方式(路由拦截、http拦截)
2018/10/24 Javascript
vue 导出文件,携带请求头token操作
2020/09/10 Javascript
web.py获取上传文件名的正确方法
2014/08/26 Python
Python对象转换为json的方法步骤
2019/04/25 Python
python3使用腾讯企业邮箱发送邮件的实例
2019/06/28 Python
Python实现某论坛自动签到功能
2019/08/20 Python
使用Fabric自动化部署Django项目的实现
2019/09/27 Python
Python大数据之使用lxml库解析html网页文件示例
2019/11/16 Python
windows系统Tensorflow2.x简单安装记录(图文)
2021/01/18 Python
国贸专业个人求职信范文
2014/01/08 职场文书
意向书范本
2014/07/29 职场文书
2015年政教主任工作总结
2015/07/23 职场文书
银行大堂经理培训心得体会
2016/01/09 职场文书
mysql查看表结构的三种方法总结
2022/07/07 MySQL