-
使用 for - in 渲染一个基于一个数组的列表 - render a list of items based on an array
指定 key 有助于提高渲染性能;据说数据变化时只会渲染更新的数据项,不会全部重新渲染
- Array
-
items 是源数据的数组
item 是迭代项/别名;可以是任意合法标识符
index 当前项的位置索引;可以是任意合法标识符;可选
<div v-for="(item, index) in items">
//
</div>
用法1:仅使用迭代数据项别名 item
<ul>
<li v-for = "item in list" :key = "item.id" > {{ item }} </li>
</ul>
用法2:同时使用数据元素对应的索引 index;使用括号()
<ul>
<li v-for = "(item, index) in list" :key = "item.id" > {{ item }} </li>
</ul>
用法3:嵌套使用;每个循环都使用独立的迭代变量;因为块级作用域,可以重名;但是不建议
<ul>
<li v-for = "(item, ind) in list" :key = "item.id" >
<p v-for = "innerItem in item" :key = "innerItem.id" > {{ innerItem }} </p>
</li>
</ul>
- Object
-
遍历一个对象的所有属性;默认显示对象的属性值
遍历的顺序会基于对该对象调用 Object.keys() 的返回值来决定
const obj = reactive({ id: 101, name: 'glpla', age: 18 })
.1个参数:表示值 value
<div v-for = "value in obj" > {{ value }} </div>
.2个参数:第一个为值 value;第二个为键 key
<div v-for = "(value, key) in obj" > {{ key }}:{{ value }} </div>
.3个参数:第一个为值 value;第二个为键 key;第三个为索引 index
<div v-for = "(value, key, index) in obj" > {{ index }}:{{ key }}:{{ value }} </div>
每个位置的参数代表 特定 的属性,如果不相使用某个参数,可以使用非标准语法 _ 表示;多个可以使用数字区分
也可以考虑使用对象的静态方法 Object.keys 或 Object.entries 来遍历对象
可读性角度出发,不建议省略;你不用它就完事了呗
<div v-for = "(_, key) in obj" > {{ key }} </div>
<div v-for = "(_1, _2, index) in obj" > {{ index }} </div>
- <template>
-
不推荐同时使用 v-if 和 v-for:二者优先级不明显
为区分结构,合理使用逻辑,特别是组合使用 v-for 和 v-if 时,可以使用 <template>;是一个不可见的包装器元素,最后渲染的结果并不会包含这个
<template> 元素
const list = ref(['home', 'product', 'about'])
const isOn = ref(true)
<template v-if = "isOn" >
<div v-for = "item in list" > {{ item }} </div>
</template>
可以在 <template> 标签上使用 v-for 来渲染一个包含多个元素的块
可以直接在组件上使用 v-for,和在一般的元素上使用没有区别
- Summary
-
v-for
key的运用
使用 <template> 处理条件渲染和列表渲染
- Homework
- [] 使用列表循环优化 类 class
中的导航/标签页
TabCard 案例
-
<template>:使用列表循环、内容绑定、属性绑定
<div class="test">
<div class="btn" :class="{ 'active': ind == index }" @click="ind = index"
v-for="(item, index) in navs"
:key="index">{{ item }}
</div>
</div>
<script>:增加一个标题数组 navs 用于导航列表循环
import { ref, watchEffect } from 'vue'
const ind = ref(0)
const navs = ref(['Lorem.', 'Labore!', 'Quos.', 'Doloremque.'])
<style>:样式不变
- [] 自动轮播 Swiper.vue
-
使用列表渲染遍历本地图片数组,如果图片分配的 ind 和轮播活动图片索引 currentInd 一致,则显示当前图片,否则隐藏
使用数字指示器
样式略
<div class="swiper">
<img :src="item" alt=""
v-for="(item, ind) in imgsUrl" :key="ind"
v-show="currentInd === ind">
<div class="indictator">
<span class="dot" :class="{ active: currentInd === ind }"
v-for="(item, ind) in imgsUrl" :key="ind">{{ ind + 1 }}</span>
</div>
</div>
import { ref, computed, onMounted, onUnmounted } from 'vue';
const imgs = ref(['coffee0.jpg', 'coffee1.jpg', 'coffee2.jpg', 'coffee3.jpg', 'coffee4.jpg'])
const currentInd = ref(0)
let timer = null
let duration = 3000
const imgsUrl = computed(() => {
return imgs.value.map(img => new URL(`../assets/imgs/${img}`, import.meta.url).href)
})
onMounted(() => {
timer = setInterval(() => {
currentInd.value++
currentInd.value = currentInd.value % imgsUrl.value.length
}, duration)
})
onUnmounted(() => {
timer && clearInterval(timer)
})
- [] 商品列表页 Goods.vue
-
列表渲染获取的商品数据,渲染多个商品项
封装组件 GoodsItem.vue 实现
<div class="goods">
<template v-if="goodsStore.goods.length">
<GoodsItem v-for="(item, ind) in goodsStore.goods" :key="item.id" :product="item" />
<footer class="f-s-s">我是有底线的~</footer>
</template>
<div v-else>商品获取失败,请刷新页面</div>
</div>
- [] 商城视图 MallView.vue - 动态处理图片资源
-
处理 assets 中的图片,动态生成 url 便于列表渲染
<div class="item" v-for="item in linksUrl">
<img :src="item.src" alt="">
<p>{{ item.title }}</p>
</div>
const links = ref([
{
id: 1,
title: '冻干咖啡',
src: 'm0.png'
},
{
id: 2,
title: '挂耳咖啡',
src: 'm1.png'
}, {
id: 3,
title: '咖啡液',
src: 'm2.png'
}, {
id: 4,
title: '咖啡豆',
src: 'm3.png'
}, {
id: 5,
title: '胶囊咖啡',
src: 'm4.png'
}
])
const linksUrl = computed(() => {
return links.value.map(item => ({
...item,
src: new URL(`../assets/${item.src}`, import.meta.url).href
}))
})
- [] 异步组件 → 标签页 - 仅使用对象的 key
-
<button v-for="(_, tab) in tabs" @click="currentTab = tab" :key="tab"
:class="['tab-button', { active: currentTab === tab }]">{{ tab }}</button>
<component :is="tabs[currentTab]" class="tab"></component>
const tabs = {
"随享瑞幸": AsyncGoods,
"颜值水杯": AsyncCup
}