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>

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

相关推荐
进击的雷神2 小时前
展位号后缀清理、详情页JS数据提取、重试机制控制、地址字段重构——美国NPE展爬虫四大技术难关攻克纪实
javascript·爬虫·python·重构
转角羊儿2 小时前
精灵图案例
开发语言·前端·javascript
大雷神2 小时前
HarmonyOS APP<玩转React>开源教程十八:课程详情页面
前端·react.js·开源·harmonyos
spencer_tseng2 小时前
secure-keyboard.js secure-keyboard.css
javascript·css
听风者一号2 小时前
cssMoudle生成器
前端·javascript·json
霍理迪2 小时前
Vue—其他指令及自定义指令
前端·javascript·vue.js
爱丽_2 小时前
Vue Router 权限路由:动态路由、导航守卫与白名单的工程落地
前端·javascript·vue.js
小江的记录本2 小时前
【Filter / Interceptor】过滤器(Filter)与拦截器(Interceptor)全方位对比解析(附底层原理 + 核心对比表)
java·前端·后端·spring·java-ee·前端框架·web
独泪了无痕2 小时前
Vue3动态组件Component的深度解析与应用
前端·vue.js·web components