Vue 中不要用 index 作为 key,key的用途

1,375 阅读3分钟

简要回答

v-for使用key

  1. key是给每一个vnode的唯一id,可以依靠key,更准确, 更的拿到oldVnode中对应的vnode节点。

  2. 用组件唯一的 id(一般由后端返回)作为它的 key,实在没有的情况下,可以在获取到列表的时候通过某种规则为它们创建一个 key,并保证这个 key 在组件整个生命周期中都保持稳定。

  3. index 作为 key,不管你数组的顺序怎么颠倒,index 都是 0, 1, 2 这样排列,导致 Vue 会复用错误的旧子节点,做很多额外的工作

  4. 千万别用随机数作为 key,不然旧节点会被全部删掉,新节点重新创建

为什么v-if使用key

提供的key属性,它可以让你自己决定是否要复用元素,key的值必须是唯一的

key的作用可以复用组件,重新渲染让生命周期再执行

详解

1、v-if中用 key 

管理可复用的元素 Vue 会尽可能高效地渲染元素,通常会复用已有元素而不是从头开始渲染。

这么做,除了使 Vue 变得非常快之外,还有一些有用的好处。例如,如果你允许用户在不同的登录方式之间切换: 

 <template v-if="loginType === 'username'">  
    <label>Username</label>  
    <input placeholder="Enter your username">  
</template>  
<template v-else>  
    <label>Email</label>  
    <input placeholder="Enter your email address">  
</template> 

那么在上面的代码中切换 loginType 将不会清除用户已经输入的内容。因为两个模版使用了相同的元素,<input> 不会被替换掉——仅仅是替换了它的 placeholder

 这样也不总是符合实际需求,所以 Vue 为你提供了一种方式来声明“这两个元素是完全独立的——不要复用它们”。只需添加一个具有唯一值的 key 属性即可: 

 <template v-if="loginType === 'username'">  
    <label>Username</label>  
    <input placeholder="Enter your username" key="username-input">  
</template>  
<template v-else>  
    <label>Email</label>  
    <input placeholder="Enter your email address" key="email-input">  
</template> 

 现在,每次切换时,输入框都将被重新渲染。注意, <label> 元素仍然会被高效地复用,因为它们没有添加 key属性

 2、v-for中的key 

 当 Vue.js 用 v-for正在更新已渲染过的元素列表时,它默认用 “就地复用” 策略。

如果数据项的顺序被改变,Vue将不是移动 DOM 元素来匹配数据项的顺序, 而是简单复用此处每个元素,并且确保它在特定索引下显示已被渲染过的每个元素。

这个类似 Vue 1.x 的 track-by="$index" 。这个默认的模式是有效的,但是只适用于不依赖子组件状态或临时 DOM 状态(例如:表单输入值)的列表渲染输出。 

为了给 Vue 一个提示,以便它能跟踪每个节点的身份,从而重用和重新排序现有元素,你需要为每项提供一个唯一 key 属性。

理想的 key 值是每项都有唯一 id。这个特殊的属性相当于 Vue 1.x 的 track-by ,但它的工作方式类似于一个属性,所以你需要用 v-bind 来绑定动态值(在这里使用简写): 

<div v-for="item in items" :key="item.id">  
    <!-- 内容 -->  
</div>

建议尽可能使用 v-for 来提供 key ,除非迭代 DOM 内容足够简单,或者你是故意要依赖于默认行为来获得性能提升。

因为它是 Vue 识别节点的一个通用机制, key 并不特别与 v-for 关联,key 还具有其他用途,我们将在后面的指南中看到其他用途。 

文章参考:juejin.cn/post/684490…