Vue的v-if和v-for放在同一个HTML元素里的坑

文章目录

环境

  • Ubuntu 24.04
  • Firefox 147.0.4 (64-bit)
  • Vue 2.7.16
  • Vue 3.5.30

背景

看下面的uni-app代码:

html 复制代码
<view v-if="ifMergeTemplate == 0" v-for="(item, i) in thMap" ......>{{ item.title }}</view>
<view v-if="ifMergeTemplate != 0" v-for="(item, i) in thMapForMerge" ......>{{ item.title }}</view>

这两行代码里,判断条件是互斥的,所以我看到代码后,随手就把第二行的 v-if="ifMergeTemplate != 0" 改成 v-else 了,没想到这一下改出了问题,所以记录下来,以后要注意避免类似的坑。

简化

一个简单页面:已有变量 x ,以及两个数组 mydata1mydata2 ,要求当 x 大于0时,显示 mydata1 的内容,否则,显示 mydata2 的内容。

判断用 v-ifv-else ,循环用 v-for

问题

Vue2代码如下:

html 复制代码
<script src="https://cdn.jsdelivr.net/npm/vue@2/dist/vue.js"></script>

<div id="app">
  <h1>{{ x }}</h1>
  <div v-if="x>0" v-for="(item, i) in mydata1" :key="item.key">{{ item.key }}: {{ item.value }}</div>
  <div v-else v-for="(item, i) in mydata2" :key="item.key">{{ item.key }}: {{ item.value }}</div>
</div>

<script type="text/javascript">
  new Vue({
    el: '#app',
    data: {
      x: 1,
      mydata1: [
        {key: 'a', value: 1},
        {key: 'b', value: 2},
        {key: 'c', value: 3}
      ],
      mydata2: [
        {key: 'd', value: 4},
        {key: 'e', value: 5}
      ]
    }
  })
</script>

x 的值为1时,页面如下:

看上去一切正常,没什么问题。

现在把 x 的值改为0,则页面如下:

可见, mydata2 的内容被打印了3次!

而且,浏览器的console报错了:

注:报错是因为key重复了,究其根源,还是因为 mydata2 被重复打印了。

分析

看下面的代码:

html 复制代码
    <div v-if="x>0" v-for="(item, i) in mydata1" :key="item.key">{{ item.key }}: {{ item.value }}</div>
    <div v-else v-for="(item, i) in mydata2" :key="item.key">{{ item.key }}: {{ item.value }}</div>

这段代码的本意 是:如果 x 大于0,则循环遍历 mydata1 ,输出item的内容。否则循环遍历 mydata2 ,输入item的内容。

然而,因为 v-if / v-elsev-for 是放在同一个HTML元素里的,这就涉及到了优先级的问题。如果前者的优先级高,那就没有问题。

但是事实上 ,在Vue2里, v-for 的优先级高于 v-if / v-else 。所以,代码逻辑就变成了:

html 复制代码
  <template v-for="(item, i) in mydata1">
    <div v-if="x>0" :key="item.key">{{ item.key }}: {{ item.value }}</div>
    <template v-else>
      <div v-for="(item, j) in mydata2" :key="item.key">{{ item.key }}: {{ item.value }}</div>
    </template>
  </template>

所以,在 x 大于0时,代码逻辑是OK的,但是当 x 小于等于0时,就会暴露出问题。

解决办法

既然 v-for 的优先级比 v-if / v-else 更高(或者你不确定哪个优先级高),那最好的解决办法就是,不要把二者放在同一个HTML元素里,而是嵌套使用。

html 复制代码
  <template v-if="x>0">
    <div v-for="(item, i) in mydata1" :key="item.key">{{ item.key }}: {{ item.value }}</div>
  </template>
  <template v-else>
    <div v-for="(item, j) in mydata2" :key="item.key">{{ item.key }}: {{ item.value }}</div>
  </template>

这样,代码结构清晰明了,阅读起来不会有误解。

其它

在Vue3里,貌似 v-if / v-else 的优先级高于 v-for ,因此对于"先判断再循环"的需求,可以直接把二者放在同一个HTML元素里:

html 复制代码
<script src="https://unpkg.com/vue@3/dist/vue.global.js"></script>

<div id="app">
  <h1>{{ x }}</h1>
  <div v-if="x>0" v-for="(item, i) in mydata1" :key="item.key">{{ item.key }}: {{ item.value }}</div>
  <div v-else v-for="(item, i) in mydata2" :key="item.key">{{ item.key }}: {{ item.value }}</div>
</div>

<script type="text/javascript">
  Vue.createApp({
    data() {
      return {
        x: 0,
        mydata1: [
          { key: 'a', value: 1 },
          { key: 'b', value: 2 },
          { key: 'c', value: 3 }
        ],
        mydata2: [
          { key: 'd', value: 4 },
          { key: 'e', value: 5 }
        ]
      }
    }
  }).mount('#app')
</script>

当然,还是推荐嵌套使用,使得代码清晰明了,消除歧义。

相关推荐
ZC跨境爬虫7 小时前
跟着 MDN 学CSS day_13 :(深入理解CSS中的元素尺寸调整)
前端·javascript·css·ui·html·tensorflow
threelab8 小时前
Three.js 加载 3D Tiles 瓦片数据 | 三维可视化 / AI 提示词
开发语言·前端·javascript·人工智能·3d·着色器
百度地图开放平台8 小时前
我用百度地图 Skills 体系重构了物流调度系统,节省了 90% 的人力
前端·github
JavaAgent架构师8 小时前
前端AI工程化(九):AI Agent平台前端架构设计
前端·人工智能
_洋8 小时前
Three.js加载 .obj文件 和 .gltf文件
开发语言·javascript·ecmascript
梦想CAD控件8 小时前
网页端对DWG图纸进行预览与批注(CAD轻量化)
java·前端·javascript
代码煮茶8 小时前
Vue3 埋点实战 | 从 0 搭建前端用户行为埋点系统
vue.js
JustNow_Man9 小时前
【opencode】安装使用daytona沙箱插件
android·java·javascript
不吃土豆的马铃薯9 小时前
Spdlog 进阶:日志基本控制、日志格式控制、异步记录器
linux·服务器·开发语言·前端·c++
wait9 小时前
Vibe Coding 开发技巧
前端·javascript·人工智能