今天写着写着,突然发现控制台里有错误:
[Vue warn]: You may have an infinite update loop in a component render function
这个问题很奇怪,之前从来没有遇到过。如果是我自己主导的项目,倒也好办,慢慢 debug 就是;偏偏在公司的项目里遇到这个问题,而公司项目的体系结构很复杂,我还没完全掌握。更恼火的是,因为体系复杂,debug 也非常困难,再加上尚无测试框架,这个难搞啊……
好死不死的,当时是下午3、4点钟,正好到了肚饿的时刻,结果又落入低血糖状态,真是屋漏偏逢连阴雨,船小又碰顶头风,饿得我脑仁生疼……
不过终于还是被我 Google + debug 出来。事实上是这样的,在 v-for
循环当中,如果用方法或者计算属性对 vm.$data 的属性进行操作,理论上,可能因为修改到循环对象,诱发无限循环。此时 Vue 就会发出警告(并不是真的已经无限循环了)。
例如这样一个组件,它里面是用 :checked + <label>
实现的一组按钮。它有以下功能:
- 为了能够分组,需要设置它们的
name
属性 - 为了能够用
<label>
控制<input>
,需要给<input>
设置id
- 按钮可以被删除
于是我选择这样做:
<template>
<div>
<template v-for="(item, index) in items">
<input type="checkbox" :name="'my-component-' + selfIndex" :id="getID">
<label :for="getID(false)">
<button type="button" @click="remove(index)">×</button>
</template>
</div>
</template>
<script>
let count = 0;
export default {
data() {
return {
selfIndex: 0,
itemIndex: 0,
}
},
methods: {
getID(increase = true) { // 注意,问题就出在这里
if (increase) {
this.itemIndex++;
}
return `my-component-${this.selfIndex}-${this.itemIndex}`;
},
},
beforeMount() {
this.selfIndex = count;
count++;
}
}
</script>
这里,为了能生成唯一 ID,我选择每次循环都对 vm.itemIndex++
,这就会出现前面说的问题,存在隐患。
解决的方案有两种,一种是把 itemIndex
也放在局部变量里,使它不直接关联在组件上;另一种则是写一个全局的唯一 ID 生成函数,然后引用进来。原理都是一样的。
这两天听评书《乱世枭雄》,学到一句话“拉屎脸朝外”,形容讲义气,不知道咋联系的……
欢迎吐槽,共同进步