写文章不容易,点个赞呗兄弟 专注 Vue 源码分享,文章分为白话版和 源码版,白话版助于理解工作原理,源码版助于了解内部详情,让我们一起学习吧
研究基于 Vue版本 【2.5.17】
如果你觉得排版难看,请点击 下面链接 或者 拉到 下面关注公众号也可以吧
初次看的兄弟可以先看 【Vue原理】NextTick - 白话版 简单了解下NextTick
好的,今天,就来详细记录 Vue 和 nextTick 的那些事
nextTick 在 Vue 中,最重要的就是~~~
协助 Vue 进行更新操作!
上篇文章
提到过,nextTick 帮助 Vue 避免频繁的更新,这里简单提一下,
每次修改数据,都会触发数据的依赖更新
也就是说数据被修改的时候,会调用一遍【引用这个数据的实例】的更新函数
那么,按道理来说,修改3次,就应该调用3遍更新函数,但是实际上只会调用一遍
比如我们使用 watch 监听 data(data 便收集了 watch 的 watcher,监听回调就是更新函数)
结果就是只打印一次
至于依赖更新,可以看下面的文章
其实,修改数据能够只更新一次,不止是 nextTick 起了作用,Vue 也做了其他处理,比如过滤实例,清空队列等等,下面就来说一下
一切先从【实例更新函数】开始
第一个要说的就是 watcher!每个实例都有一个 watcher,然后 watcher 保存着实例的更新函数
每个实例都会通过 new Vue 生成的,所以会有一个专属的 watcher
更新函数被保存在 watcher.getter 上
function Vue(){
....
new Watcher(vm, 实例更新函数)
}
function Watcher(vm, expOrFn) {
this.getter = expOrFn;
};
Watcher.prototype.get = function() {
this.getter.call(vm, vm);
};
Watcher.prototype.update = function() {
queueWatcher(this);
};
Watcher.prototype.run = function() {
this.get();
};
我们知道, Vue 的 data 是响应式的,就是通过 Object.defineProperty 设置 get 和 set
当数据被修改的时候, set 函数被触发,函数内部会通知所有的实例进行更新(就是调用每个实例的 watcher.update 方法)
具体可以看这个
那么我们现在的重点就在 watcher.update 上了,看看上面的 Watcher 代码
出现了一个 queueWatcher 的东西
速度看源码!
var queue = [];
var has = {};
var index = 0;
var flushing = false;
var waiting= false;
function queueWatcher(watcher) {
var id = watcher.id;
// 如果是同一个 Vue 实例,就不要重复添加了
if (has[id] == null) {
// 这个实例已经被标记了
has[id] = true;
// 如果没有在运行,那么直接放入队列
if (!flushing) {
queue.push(watcher);
} else {
// if already flushing, splice the watcher based on its id
// if already past its id, it will be run next immediately.
var i = queue.length - 1;
// 跳过所有比我大的
while (i > index && queue[i].id > watcher.id) {
i--;
}
// 最后放在队列中,一个比我老的 watcher 后面
queue.splice(i + 1, 0, watcher);
}
// 在 flushSchedulerQueue 执行之后设置为 false
if (!waiting) {
waiting = true;
nextTick(flushSchedulerQueue);
}
}
}
先说说其中涉及的几个变量
是一个对象,用来过滤watcher。
当这个watcher 已经调用过更新函数,那么就在 has 中标记这个 id
也就是,你同时间调用多次 watcher.update ,其实只有第一次调用有用,后面的都会被过滤掉
一个数组,watcher 更新队列,存放需要更新的 watcher
flushSchedulerQueue
watcher 更新队列执行函数,下面有讲到
为 true 表示已经把 【watcher 更新队列执行函数】 注册到宏微任务上了(或者说存放进 callbacks 中)。
正在等待JS栈为空后,就可以执行更新。直到所有watcher 更新完毕,才重置为 false
为 true 表示 watcher 更新队列正在执行更新(就是开始遍历 watcher 队列,逐个调用 watcher 更新了)
直到所有watcher 更新完毕,才重置为 false
queueWatcher 源码不算很复杂,主要做两件事
1、处理watcher 更新队列 queue
2、注册 【watcher 更新队列 执行函数】进宏微任务
当 flushing 为 false时,表示 queue 还没有开始遍历执行,直接 push
当 flushing 为 true,表示 queue 已经开始遍历,执行其中的 watcher 更新了
然后,做了一个很特殊的插入操作(为了方便看,把上面的源码截取了)
我还是没有看懂这是为什么? 直到我看到了 flushSchedulerQueue 的 源码!
因为在 flushSchedulerQueue 执行的时候(此时设置了 flushing = true),内部把 queue 升序排列了!
所以在 flushing 的时候,queue已经是有序状态,中途进来的 watcher,当然也要按顺序来
所以,这一段的作用就是给 新来的 watcher 排序!
其中 index 表示 现在正遍历到第几个 watcher(在 flushSchedulerQueue 中设置)
所以,也必然是排到已经执行过的 watcher 后面的(不然就遍历不到这个watcher 了啊)
已经讲到 flushSchedulerQueue 了,他就是 注册宏微任务的异步回调
直接存放进 异步任务队列 callbacks 中的
关于 nextTick 的 异步任务队列 ,可以看
接下来,就看 flushSchedulerQueue
function flushSchedulerQueue() {
flushing = true;
var watcher;
// 升序排列
queue.sort(function(a, b) {
return a.id - b.id;
});
for (index = 0; index < queue.length; index++) {
watcher = queue[index];
has[watcher.id] = null;
watcher.run();
}
// 所有watcher 完成更新,重置状态
queue.length = 0;
has = {};
waiting = flushing = false;
}
1、升序排列 watcher 更新队列
2、遍历 watcher 更新队列,然后逐个调用 watcher 更新
3、watcher 更新队列执行完毕,重置状态
其他我都看得明白,唯独我不懂一个问题
为什么要把 queue 按照 watcher.id 升序排列??
首先,watcher.id 越大,表示这个 watcher 越年轻,实例是越后面生成的
This ensures that:
我只挑一点
为什么先更新父组件,再更新子组件,我还是想不通啊?
个人认为,因为父组件跟子组件是有联系的,什么联系呢?
比如 props
当 父组件传给子组件的数据变化的时候,父组件需要把 变化后的数据 传给 子组件,子组件才能知道数据变了
那么 子组件才能更新组件内使用 props 的地方
所以,父组件必须先更新,把最新数据传给 子组件,子组件再更新,此时才能获取最新的数据
不然你子组件更新了,父组件再传数据过来,那就不会子组件就不会显示最新的数据了啊
至于 父组件更新时怎么传 数据给子组件的?
数据变化,通知 watcher 更新,watcher.update
queueWatcher 把 watcher 添加进 【queue 更新队列】
把 flushSchedulerQueue 注册进宏微任务
JS 主栈执行完,开始执行异步代码
flushSchedulerQueue 遍历 queue ,逐个调用 watcher 更新
完成更新