Vue 2 vs Vue 3:v-if 和 v-for 的差异

在 Vue 2 和 Vue 3 中,v-if 和 v-for 是用于条件渲染和列表渲染的两个常用指令。虽然两个版本中它们的基本用法一致,但在某些情况下有差异和注意事项。

同时,这也是比较常见的 Vue 面试题之一。

1. 基本用法

1、v-if 用于条件渲染,当条件为 true 时,元素才会被渲染。

html 复制代码
<!-- Vue 2 和 Vue 3 中的用法一致 -->
<div v-if="isVisible">This is visible</div>

2、v-for 用于渲染列表,通过迭代数组或对象的每一项生成一组元素。

html 复制代码
<!-- Vue 2 和 Vue 3 中的用法一致 -->
<ul>
  <li v-for="item in items" :key="item.id">{{ item.name }}</li>
</ul>

2. v-if 和 v-for 结合

在 Vue 2 和 Vue 3 ,如果在同一个元素上同时使用 v-if 和 v-for,它们的执行顺序是不同的。

-Vue 2,v-for 的优先级高于 v-if,这意味着即使某个列表项不满足 v-if 条件,它也仍然是先被 v-for 迭代,然后才根据 v-if 的条件决定是否渲染。

html 复制代码
<!-- Vue 2 中的代码 -->
<template>
  <ul>
    <li v-for="item in items" v-if="item.isVisible" :key="item.id">
      {{ item.name }}
    </li>
  </ul>
</template>
<script>
export default {
  data() {
    return {
      items: [
        { id: 1, name: 'Item 1', isVisible: true },
        { id: 2, name: 'Item 2', isVisible: false },
        { id: 3, name: 'Item 3', isVisible: true },
      ]
    };
  }
};
</script>
  • Vue 3,v-if 的优先级高于 v-for,这意味着在迭代之前,首选会判断是否满足 v-if 的条件,如果条件为 false,则不会执行 v-for。
html 复制代码
<!-- Vue 3 中的代码 -->
<template>
  <ul>
    <li v-if="shouldRender" v-for="item in items" :key="item.id">
      {{ item.name }}
    </li>
  </ul>
</template>
<script>
export default {
  data() {
    return {
      shouldRender: false, // 控制是否渲染整个列表
      items: [
        { id: 1, name: 'Item 1', isVisible: true },
        { id: 2, name: 'Item 2', isVisible: false },
        { id: 3, name: 'Item 3', isVisible: true },
      ]
    };
  }
};
</script>

Vue 3 认为 Vue 2 设计的优先级有问题,虽然不报错,但是不建议使用。因此在 Vue 3 中,直接设计 v-if 的优先级高于 v-for。(先执行 v-if,然后执行 v-for)

3. 改进

在 Vue 3 中,官方建议,避免在同一个元素上同时使用 v-if 和 v-for,而是将它们分开处理。不仅可以提升代码的可读性,还能避免潜在的逻辑错误。

1、Vue 2 使用计算属性分离逻辑

Vue 2 中,v-for 的优先级高于 v-if,导致每一次循环时判断,修改数据之后,也要重新循环所有项,会大大耗损效率。

html 复制代码
<template>
  <ul>
    <!-- 只使用 v-for,因为计算属性已经过滤了不可见的项 -->
    <li v-for="item in visibleItems" :key="item.id">
      {{ item.name }}
    </li>
  </ul>
</template>
<script>
export default {
  data() {
    return {
      items: [
        { id: 1, name: 'Item 1', isVisible: true },
        { id: 2, name: 'Item 2', isVisible: false },
        { id: 3, name: 'Item 3', isVisible: true },
      ]
    };
  },
  computed: {
    // 计算属性:只返回 isVisible 为 true 的项
    visibleItems() {
      return this.items.filter(item => item.isVisible);
    }
  }
};
</script>

2、Vue 3 分离逻辑到外部容器

html 复制代码
<template>
  <!-- 将 v-if 移动到外部容器 -->
  <div v-if="shouldRender">
    <ul>
      <li v-for="item in items" :key="item.id">
        {{ item.name }}
      </li>
    </ul>
  </div>
</template>

<script>
export default {
  data() {
    return {
      shouldRender: false, // 控制是否渲染整个列表
      items: [
        { id: 1, name: 'Item 1', isVisible: true },
        { id: 2, name: 'Item 2', isVisible: false },
        { id: 3, name: 'Item 3', isVisible: true },
      ]
    };
  }
};
</script>

4. 注意点

4.1 v-for 的 key

在 Vue 中,key 属性是 Vue 用于标识和追踪每个列表项的唯一标识符,用于在渲染列表时帮助 Vue 追踪每个元素的身份(确定哪些元素被改变、添加或移除),从而在更新DOM 时优化性能。

也就是,在数据变化时,Vue 能够精确地找到被改变的元素,并且只更新这些元素,避免了整个列表的重新渲染。

html 复制代码
<!-- 假设 items 是一个数组,每个 item 都有唯一的 id -->
<div v-for="item in items" :key="item.id">
  {{ item.name }}
</div>
4.2 面试题:为什么不推荐使用索引作为 key

1、列表项顺序变化:当顺序发生变化时,使用索引作为 key 可能导致渲染错误。Vue 使用 key 识别哪个元素是新的,哪个是旧的,以便在 DOM 中高效的更新。如果使用索引,当列表项顺序变化时,Vue 可能无法正确地识别出元素的移动,而可能会错误地删除或重新创建元素。

2、动态添加或删除列表项:当列表项被动态添加或删除时,使用索引也可能导致问题。例如,如果从列表中删除一个元素,所有后续元素的索引都会改变,这可能会导致 Vue 错误地重新渲染整个列表。

3、性能问题:Vue 使用 key 以便最小化 DOM 操作,更新那些真正改变的 DOM 元素。如果 key 不唯一或者不稳定,Vue 可能需要执行更多的 DOM 操作来更新列表,导致性能降低。

4、组件复用问题:如果列表项是一个 Vue 组件,Vue 会尝试重用具有相同 key 值的组件实例,如果 key 不稳定,可能导致组件状态的错误或丢失。

5、可预测性:使用一个稳定的、可预测的 key 可以提高 Vue 的更新算法的效率。每个节点的 key 唯一,以便 Vue 可以准确跟踪,从而重用和重新排序现有的元素,而不是重新创建它们。

因此推荐使用数据项的唯一标识符(如 ID)作为 key,这样可以确保在数据变化时,Vue 能够准确追踪和更新 DOM 元素。

html 复制代码
<div v-for="item in items" :key="item.id">
  {{ item.name }}
</div>
相关推荐
初遇你时动了情6 分钟前
react 项目打包二级目 使用BrowserRouter 解决页面刷新404 找不到路由
前端·javascript·react.js
乔峰不是张无忌33025 分钟前
【HTML】动态闪烁圣诞树+雪花+音效
前端·javascript·html·圣诞树
鸿蒙自习室33 分钟前
鸿蒙UI开发——组件滤镜效果
开发语言·前端·javascript
m0_7482507440 分钟前
高性能Web网关:OpenResty 基础讲解
前端·openresty
前端没钱1 小时前
从 Vue 迈向 React:平滑过渡与关键注意点全解析
前端·vue.js·react.js
汪洪墩1 小时前
【Mars3d】设置backgroundImage、map.scene.skyBox、backgroundImage来回切换
开发语言·javascript·python·ecmascript·webgl·cesium
NoneCoder1 小时前
CSS系列(29)-- Scroll Snap详解
前端·css
无言非影1 小时前
vtie项目中使用到了TailwindCSS,如何打包成一个单独的CSS文件(优化、压缩)
前端·css
我曾经是个程序员2 小时前
鸿蒙学习记录
开发语言·前端·javascript
顽疲2 小时前
springboot vue 会员收银系统 含源码 开发流程
vue.js·spring boot·后端