就目前所了解的情况,key的作用有以下这些。
场景一大同小异司空见惯,场景二是下面这样的:
<div :key="rerender">
<span>Hello vue.js !</span>
<complexComponent :propObj="propObj" :propArr="propArr" ></complexComponent>
</div>
refresh(){
this.rerender = + new Date();
}
那么vue中key的相关知识点到底是怎样的呢?
<ul>
<li v-for="item in items" :key="item.id">...</li>
</ul>
<transition>
<span :key="text">{{ text }}</span>
</transition>
text发生变化时,<span>会被replaced,而不会patched,因此transition会被触发。
我的理解:
text变化时,span的key发生了变化,也就是说曾经拥有了旧key的span不再出现了,当拥有新值的text作为key时,拥有了新key的span出现了,那么旧key span会被移除,旧transition也会移除,新key span触发渲染,新transition触发。
结合官方API的知识点,现在再来回顾文章开头提出的场景。
答案:
<div :key="rerender">
<span>Hello Vue.js !</span>
<complexComponent :propObj="propObj" :propArr="propArr" ></complexComponent>
</div>
refresh(){
this.rerender = + new Date();
}
答案:
思考:
由于Vue.js的obj和arr存在无法检测到数据变化的情况,obj是属性的新增和删除(原因是新增和删除都没有触发setter,watcher未告诉外界更新),arr则是数组内元素重新赋值或者修改length属性(原因是没有使用改变数组本身的方法,没有触发数组原型链拦截器,watcher未告诉外界更新)。
所以!通过赋予新key的方式,移除旧key div,渲染新key div,propObj和propArr在complexComponent组件内会重新触发一次生命周期,做一次重新渲染。此时父组件的propObj和propArr js变量其实已经获取到新值了,只是没有触发DOM也好,VNode也好的重新渲染。需要通过刷新key去force update,说到forceUpdate,可以通过$forceUpdate()去手动强制更新DOM。
场景:父组件修改传递给子组件的数据,数组数据的更新没有按照this.$set去更新。该怎么办?
this.productImages.forEach((product) => {
if (product.productId in this.productsState) {
product.status = this.productsState[product.productId];
}
});
不使用this.$set去赋值数据的不能rerender的原因是什么?
在Vue.js中,对Array的变化侦测是通过拦截原型的方式实现的。也就通过对push,pop,shift,unshift,splice,sort,reverse,fill,copyWithin去改变数组自身内容的方法做拦截,从而响应。而product.status = this.productsState[product.productId];没有触发任何改变数组自身的被监听的方法,因此不会rerender。
加在this.productImages的父元素上就好。
若不涉及数据传递,也可以直接加在需要更新的element上。
现在是粗暴的+new Date()时间戳做key值的。
也可以用双向绑定的值作为key值,保证新旧key值不同就行。
vue.js的虚拟DOM算法,在更新vNode时,需要从旧vNode列表中查找与新vNode节点相同的vNode进行更新,如果这个过程设置了属性key,过程就会快很多。
其他具体见上文。
只能在父组件调用这个方法,手动通知vue实例重新渲染。
// $forceUpdate源码
Vue.prototype.$forceUpdate = function () {
const vm: Component = this
if (vm._watcher) {
vm._watcher.update()
}
}
// update源码
/**
* Subscriber interface.
* Will be called when a dependency changes.
*/
update () {
/* istanbul ignore else */
if (this.lazy) {
this.dirty = true
} else if (this.sync) {
this.run()
} else {
queueWatcher(this)
}
}
product.status = this.productsState[product.productId];以后,其实此时dep已经发生变化了,但是Vue.js数组响应式的实现由于是拦截原型链方法的方式,没有检测到这个变化,所以不会自动rerender,没有触发update。因此我们通过$forceUpdate的方式,调用包含dep的watcher上的update方法,从而做到rerender。
不可以。
因为dep是父组件的watcher和dep,并不是子组件,是父组件的this.productImages没有被检测到并实时更新,并不是子组件的问题。
引用 Vue 官方文档的原话:Vue 会尽可能高效地渲染元素,通常会复用已有元素而不是从头开始渲染。这样容易导致一些问题。因为类似 <input> , <select> , <textarea>
尤大在vue 2.x的文档中明确指出:建议尽可能在使用 v-for 时提供 key attribute,除非遍历输出的 DOM 内容非常简单,或者是刻意依赖默认行为以获取性能上的提升。
内容以共享、参考、研究为目的,不存在任何商业目的。其版权属原作者所有,如有侵权或违规,请与小编联系!情况属实本人将予以删除!