【发布时间】:2021-03-31 12:32:02
【问题描述】:
这是我为调查 vue3 中列表的不必要节点重新渲染而进行的一个小测试(vue2 具有相同的行为):https://kasheftin.github.io/vue3-rerender/。这是源代码:https://github.com/Kasheftin/vue3-rerender/tree/master。
我试图理解为什么 vue 在某些情况下会重新渲染 v-for 中已经渲染的节点。我知道(并将在下面提供)一些避免重新渲染的技术,但对我来说,理解理论至关重要。
对于测试,我添加了一个虚拟 v-test 指令,该指令仅在触发 mount/beforeUnmount 挂钩时记录。
测试 1
<div v-for="i in n" :key="i">
<div>{{ i }}</div>
<div v-test="log2">{{ log(i) }}</div>
</div>
结果:当 n 增加时,所有节点都重新渲染。为什么?如何避免?
测试 2
Test2.vue:
<RerenderNumber v-for="i in n" :key="i" :i="i" />
RerenderNumber.vue:
<template>
<div v-test="log2">{{ log() }}</div>
</template>
结果:它工作正常。将内部内容从 test1 移动到单独的组件可以解决此问题。为什么?
测试 3
<RerenderObject v-for="i in n" :key="i" :test="{ i: { i: { i } } }" />
结果:不必要的重新渲染。在将对象发送到某个子组件之前,似乎不允许在循环中动态构造对象,可能是因为 JavaScript 中的 {} != {}。
测试 4
<template>
<RerenderNumberStore v-for="item in items" :key="item.id" :item="item" />
</template>
<script>
export default {
computed: {
items () {
return this.$store.state.items
}
},
methods: {
addItem () {
this.$store.commit('addItem', { id: this.items.length, name: `Item ${this.items.length}` })
}
}
}
</script>
这里使用了最简单的 vuex 存储。它工作正常 - 尽管 item prop 是一个对象,但没有不必要的重新渲染。
测试 5
<RerenderNumberStore v-for="item in items" :key="item.id" :item="{ id: item.id, name: item.name }" />
与测试 4 相同,但重新构建了 item 道具 - 我们得到了不必要的重新渲染。
测试 6
Test6.vue:
<RerenderNumberStoreById v-for="item in items" :key="item.id" :item-id="item.id" />
RerenderNumberStoreById.vue:
<template>
<div v-test="log">{{ item.name }}</div>
</template>
<script>
export default {
props: ['itemId'],
computed: {
item () { return this.$store.state.items.find(item => item.id === this.itemId) }
}
}
</script>
结果:不必要的重新渲染。为什么?我找不到行为与测试 4 不同的任何原因。这对我来说不太清楚 - 当新项目添加到 items 数组时,计算的项目不会以任何方式更改。它返回相同的对象。它必须被缓存,与之前的值匹配,并且不会触发 DOM 中的任何更新。
【问题讨论】:
标签: vue.js rendering vuejs3 virtual-dom