从lifecycleMixin中挂载到Vue构造函数的prototype属性上。
export function lifecycleMixin(Vue){
Vue.prototype.$forceUpdate = function(){
//做点什么
}
Vue.prototype.$destory = function(){
//做点什么
}
}
从renderMixin中挂载到Vue构造函数的prototype属性上的。
export function renderMixin(Vue){
Vue.prototype.$nextTick = function(fn){
//做点什么
}
}
在跨平台的代码中挂载到Vue构造函数的prototype属性上的。
迫使Vue.js实例重新渲染。注意它仅仅影响实例本身以及插入插槽内容的子组件,而不是所有子组件。
1、只需要执行watcher的update方法,就可以让实例重新渲染。
2、Vue.js的每一个实例都有一个watcher。当状态发生改变时,会通知到组件级别,然后组件内部使用虚拟DOM进行更详细的重新渲染操作。
3、事实上,组件就是Vue.js实例,所以组件几倍的watcher和Vue.js实例上的watcher说的是同一个watcher。
4、手动执行实例watcher的update方法,就可以使Vue.js实例重新渲染。
Vue.prototype.$forceUpdate = function(){
const vm = this;
if(vm._watcher){
vm._watcher.update();
}
}
5、vm._watcher就是Vue.js实例的watcher,每当组件内依赖的数据发生变化时,都会自动触发Vue.js实例中_watcher的update方法。
6、重新渲染的实现原理并不难,Vue.js的自动渲染通过变化侦测来侦测数据,即当数据发生变化时,Vue.js实例重新渲染。而vm.$forceUpdate是手动通知Vue.js实例重新渲染。
完全销毁一个实例,它会清理该实例与其他实例的连接,并解绑其全部指令及监听器,同时会触发beforeDestory和destroyed的钩子函数。
(2)这个方法并不是很常用,大部分场景下并不需要销毁组件,只需要使用v-if或则v-for等指令以数据驱动的方式控制子组件的生命周期即可。
Vue.prototype.$destory = function(){
const vm = this;
if(vm._isBeingDestroyed){
return;
}
callHook(vm,"beforeDestroy");
vm._isBeingDestroyed = = true;
}
1、为了防止vm.$destroy被反复执行,先对属性_isBeingDestroyed进行判断,如果它为true,说明Vue.js实例正在被销毁,直接使用return语句退出函数执行逻辑。因为销毁只需要销毁一次即可,不需要反复销毁。
2、然后调用callHook函数触发beforeDestroy的钩子函数(callHook会触发参数中提供的钩子函数)。
首先,需要清理当前组件与父组件之间的连接。组件就是Vue.js实例,所以要清理当前组件与父组件之间的连接,只需要将当前组件实例从父组件实例的$children属性中删除即可。
说明:Vue.js实例的$children属性存储了所有子组件
const parent = vm.$parent;
if(parent && !parent._isBeingDestroyed && !vm.$options.abstract){
remove(parent.$children,vm)
}
1、如果当前实例有父级,同时父级没有被销毁且不是抽象组件,那么将自己从父级的子列表中删除,也就是将自己的实例从父级的$children属性中删除。
2、事实上,子组件在不同父组件中是不同的Vue.js实例,所以一个子组件实例的父级只有一个,销毁操作也只需要从父级的子组件列表中销毁当前这个Vue.js实例。
export function remove(arr,item){
if(arr.length){
const index = arr.indexOf(item);
if(index>-1){
return arr.splice(index,1);
}
}
}
1、父子组件间的链接断掉之后,需要销毁实例上的所有watcher,也就是说需要将实例上所有的依赖追踪断掉。
2、状态会收集一些依赖,当状态发生改变时会向这些依赖发送通知,而被收集的依赖就是watcher实例。因此,当Vue.js实例被销毁时,应该将实例所监听的状态都取消掉,也就是从状态的依赖列表中将watcher移除。
3、watcher的teardown方法,它的作用是从所有依赖项的Dep列表中将自己移除。即只要执行这个方法,就可以断掉这个watcher所监听的所有状态。
4、断掉Vue.js实例自身的watcher实例监听的所有状态。
if(vm._watcher){
vm._watcher.teardown();
}
5、执行了组件自身的watcher实例的teardown方法,从所有依赖项的订阅列表中删除watcher实例。删除之后,当状态发生变化时,watcher实例就不会再得到通知。
6、vm._watcher来源
当执行new Vue()时,会执行一系列初始化操作并渲染组件到实体上,其中就包括vm._watcher的处理
7、从Vue.js2.0开始,变化侦测的粒度调整为中等粒度,它只会发送通知到组件级别,然后组件使用虚拟DOM进行重新渲染。组件其实就是Vue.js实例。
8、怎么通知到组件级别
在Vue.js实例上,有一个watcher,也就是vm._watcher,它会监听这个组件中用到的所有状态,即这个组件内用到的所有状态的依赖列表中都会收集到vm._watcher。当这些状态发生变化时,也都会通知vm._watcher,然后这个watcher再调用虚拟DOM进行重新渲染。
1、只从状态的依赖列表中删除Vue.js实例上的watcher实例是不够的。Vue.js提供了vm.$watch方法,它允许用户监听某个状态。因此,还需要销毁用户使用vm.$watch所创建的watcher实例。
2、从状态的依赖列表中销毁用户创建的watcher实例和销毁Vue实例上的watcher实例相同,只需要执行watcher的teardown方法。
3、问题:如何知道用户创建了多少个watcher?
1)Vue.js的解决方案是执行new Vue()时,在初始化的流程中,在this上添加一个_watchers属性
vm._watchers = [];
2)每当创建watcher实例时,都会将watcher实例添加到vm._watchers中
export default class Watcher{
constructor(vm,expOrFn,cb){
vm._watchers.push(this);
}
}
4、只需要遍历vm._watchers并依次执行每一项watcher实例的teardown方法,就可以将watcher实例从它所监听的状态的依赖列表中移除。
let i = vm._watchers.length;
while(i--){
vm._watchers[i].teardown();
}
vm._isDestroyed = true;
vm._patch_(vm._vnode,null)
callHook(vm,'destroyed')
vm.$off()
Vue.prototype.$destory = function(){
const vm = this;
if(vm._isBeingDestroyed){
return;
}
callHook(vm,"beforeDestroy");
vm._isBeingDestroyed = = true;
const parent = vm.$parent;
if(parent && !parent._isBeingDestroyed && !vm.$options.abstract){
remove(parent.$children,vm);
}
if(vm._watcher){
vm._watcher.teardown();
}
let i = vm._watchers.length;
while(i--){
vm._watchers[i].teardown();
}
vm._isDestroyed = true;
vm._patch_(vm._vnode,null)
callHook(vm,'destroyed')
vm.$off();
}