vue的就地更新
Vue中的就地更新到底是怎么回事,为什么会存在就地更新的现象?
注意下面的例子,使用v-for指令时,没有绑定key值,才有就地更新的现象,因为Vue默认按照就地更新的策略来更新v-for渲染的元素列表
下面的例子很简单,就是循环遍历数据list
,当li
标签被点击的时候就进行删除操作。现在我们关心的并不是如何实现功能,而是点击删除按钮时,整个Elements
DOM结构树是如何变化的。
按照常理来说,如果我们点击第一个li
标签的删除按钮,那么就应该删除第一个li
标签,点击第二个删除按钮,就删除第二个li
标签,依次类推。但实际上Vue并不是这样做的,那Vue是怎么做的呢?
比如说,我现在点击第二个li
标签的删除按钮,第二个li
标签的内容是item-2
。实际上Vue是先将第三个li
标签的内容item-3
,移动到item-2
的<span></span>
标签内部,然后再删除内容是item-3
的li
标签。换句话说,就是点击第二个删除按钮,删除的并不是第二个li
标签,其实删除的是第三个li
标签,Vue只是将第三个li
标签中<span></span>
标签的内容item-3
,移动到第二个li
标签中的<span></span>
标签内部。
const App = {data() {return {list: [{id: 1,value: 'item-1'},{id: 2,value: 'item-2'},{id: 3,value: 'item-3'}]}},template:`<ul><li v-for="(item, index) of list"><span>{{ item.value }}</span><button @click="handleClickDel(index)">Delete</button></li></ul>`,methods: {handleClickDel(index) {this.list.splice(index,1);}}
}
没有点击item-2时的dom结构
点击item-2的delete按钮时,dom树的变化
点击item-2
的Delete
按钮时,我们发现DOM结构树发生了更改。注意Elements
背景色是紫色的标签,背景是紫色说明标签内容发生了变化。那么我们之前说,Vue此时删除的并不是第二个li
标签,只是将item-3
字符串移动到第二个li
标签中的<span></span>
标签中,删除第三个li
标签。所以我们从图中可以看到,第二个li
标签内容发生变化,而第三个li
标签被删除。这种现象就称为就地更新。
就地更新的优势
上面叙述了就地更新的现象,那为什么Vue中存在就地更新的现象呢?它的优势又是什么呢?我们再来看一个例子吧。
我们首先看模板template
中的两个<div></div>
标签有什么共同点呢?两个<div></div>
标签中都拥有标签<a href="#"></a>
,按照常理来说,如果变量isLogin
返回的是Truthy(真值)的时候,那么第一个<div></div>
标签内部的span、a
标签都会被渲染;当isLogin
返回的是Falsy(虚值)的时候,那么第二个<div></div>
标签内部的两个a
标签也都会被渲染。
但是实际上并不是这样滴,Vue拥有就地更新的现象。所以点击<a href="javascript:;" @click="isLogin = true">登录</a>
的时候,Vue会将<a href="#">注册</a>
标签内部的字符串注册文字改为xiaoming,对这个<a href="#"></a>
标签进行复用🤔,而不是重新渲染一个新的DOM元素。
我们通过这个例子就能看出Vue中就地更新的优势,首先就地更新是非常高效的,能够复用其它的DOM元素,减少DOM操作,减少DOM元素的重绘与回流,减少浏览器的性能消耗。
const App = {data() {return {isLogin: false}},template:`<divv-if="isLogin"><span>欢迎您~</span><a href="#"> xiaoming </a></div> <div v-else><a href="javascript:;" @click="isLogin = true">登录</a><a href="#">注册</a></div>`
}
就地更新的问题
默认模式是高效的,但是只适用于列表渲染输出的结果不依赖子组件或者临时的DOM状态(例如表单输入值)的情况。
我们如何理解“**只适用于列表渲染输出的结果不依赖子组件或者临时的DOM状态(例如表单输入值)的情况”**这句话呢?我们来通过下面的例子进行解释。
就地更新遇到临时的dom状态
下面的例子中,(图1)是DOM结构树初始化的状态。当我手动往input
框输入值后,点击item-2
的Detele
按钮,此时会出现(图2)的状态,我们发现Vue的的确确是按照就地更新的策略进行的渲染列表。但是我们输入框输入的值并没有进行更新。是因为我手动输入值的状态是临时DOM状态,Vue没有办法判断节点内部这个临时DOM状态有什么用,因此Vue并不去会跟踪这个状态。所以就地更新遇见临时DOM状态就会出现(图2)中的问题。
const App = {el:'#app',data () {return {list: [{id: 1,value: 'item-1'},{id: 2,value: 'item-2'},{id: 3,value: 'item-3'}]}},template:`<ul><li v-for="(item, index) of list" :key="item.id"><span> {{ item.value }} </span><input type="text" /><button @click="deleteItem(index)">DELETE</button></li></ul>`,methods: {deleteItem(index) {this.list.splice(index,1);} }
}
(图1)
(图2)
当然上面例子相当于这样的写法,此时tempArr
变量是固定的,是不变化的;因为tempArr
是不变化的,那么input
标签内部绑定的value
值是没有办法更新的,而碰上Vue当中的就地更新的策略就出现(图2)中同样的问题。
const App = {data() {return {tempArr: [1, 2, 3],list:[{id: 1,value: 'item-1'},{id: 2,value: 'item-2'},{id: 3,value: 'item-3'}]}},template: `<div><ul><li v-for="(item,index) of list"><span> {{ item.value }} </span><input type="text" :value="tempArr[index]" /><button @click="deleteItem(index)">DELETE</button></li></ul></div>`,methods: {deleteItem(index) {this.list.splice(index, 1);}}
}
就地更新遇到子组件
Vue文档中指出就地更新**只适用于列表渲染输出的结果不依赖子组件,**也就是说你模板中列表渲染的结果不依赖子组件的时候就地更新策略不会出现问题。但是如果你模板中列表渲染的结果依赖子组件的时候,就地更新策略就会出现问题。因为子组件内部可能会有复杂的逻辑,所以Vue监控不了子组件内部的数据。
例如下面中的例子,列表渲染输出的结果依赖子组件MyComponent
(需要子组件MyComponent
),然后Vue中v-for
指令默认按照就地更新策略进行渲染,所以出现(图3)中出现的问题。
const MyComponent = {props: {num: Number,},template: `<span> {{ num }} </span>`
}
const App = {components: {MyComponent},data() {return {list: [{id: 1,value: 'item-1'},{id: 2,value: 'item-2'},{id: 3,value: 'item-3'}],tempArr:[1, 2, 3]}},template: `<div><ul><li v-for="(item, index) of list"><span> {{ item.value }} - </span><my-component :num="tempArr[index]"></my-component><input type="text" :value="tempArr[index]" /><button @click="deleteItem(index)">DELETE</button></li></ul></div>`,methods: {deleteItem(index) {this.list.splice(index, 1);}}
};
(图3)
解决就地更新遇到的问题
解决Vue中就地更新的方法就是给循环的每一个标签绑定一个唯一的key
值,并且key
值不能够变更。添加key
值之后就没有就地更新的现象出现了,因为添加唯一key
值后,Vue后面的驱动能够追踪绑定的元素。
下面的例子中我们在v-for
指令下绑定唯一的key
值为item.id
,此时按照常理来说已经可以解决就地更新遇见的问题,但是发现还是没有效果,v-for
指令渲染列表时依旧使用的是就地更新的策略。这又是为什么呢?
原因在于<my-component :num="tempArr[index]" />
和<input type="text" :value="tempArr[index]" />
标签中,tempArr[index]
属性的index
属性来源于v-for
指令,因为这个index
值会随着绑定的数据list
长度的变化而变化,所以导致绑定的元素无法对应上,所以还是会执行就地更新的策略。
这也是为什么不推荐v-for
指令中,绑定key
值不要绑定index
的原因,因为你的增加、删除操作会导致index
值的变化,而如果绑定的key
值在渲染之后还会不断的变化,那么导致在Vue底层中判断新老节点的结果一致(底层源码中通过a.key ===b.key判断key
值),如果结果一致的话,就执行就地更新的策略。如果绑定的key
值是静态的,那么新节点绑定的key
值于老节点绑定的key
值肯定不同,如果底层判断新老节点的结果不一致的话,会进行打补丁patch的其它逻辑判断,不会执行就地更新的逻辑,就能够追踪每一个绑定的节点。
const MyComponent = {props: {num: Number,},template: `<span> {{ num }} </span>`
}
const App = {components: {MyComponent},data() {return {list: [{id: 1,value: 'item-1'},{id: 2,value: 'item-2'},{id: 3,value: 'item-3'}],tempArr:[1, 2, 3]}},template: `<div><ul><li v-for="(item, index) of list":key="item.id"><span> {{ item.value }} - </span><my-component :num="tempArr[index]"></my-component><input type="text" :value="tempArr[index]" /><button @click="deleteItem(index)">DELETE</button></li></ul></div>`,methods: {deleteItem(index) {this.list.splice(index, 1);}}
};
通过上述的逻辑,我们修改模板template
的代码,将绑定过动态index
值的地方都换成静态的属性item.id
。按照分析,v-for
指令渲染列表时候就不会执行就地更新的策略,Vue也就能够追踪绑定的元素。从(图4)也可以看出的的确确没有执行就地更新的策略。
template: `<div><ul><li v-for="(item, index) of list":key="item.id"><span> {{ item.value }} - </span><my-component :num="item.id"></my-component><input type="text" :value="item.id" /><button @click="deleteItem(index)">DELETE</button></li></ul></div>`,
(图4)