this.$nextTick(cb)
一、简化图

二、异步更新
Vue 的异步更新机制的核心是利用了浏览器的异步任务队列来实现的,首选微任务队列,宏任务队列次之。
当响应式数据更新后,会调用 dep.notify 方法,通知 dep 中收集的 watcher 去执行 update 方法,watcher.update 将 watcher 自己放入一个 watcher 队列(全局的 queue 数组)。如果同一个watcher被多次触发,只会被推入到队列中一次。
然后通过 nextTick 方法将一个刷新 watcher 队列的方法(flushSchedulerQueue)放入一个全局的 callbacks 数组中。
如果此时浏览器的异步任务队列中没有一个叫 flushCallbacks 的函数,则执行 timerFunc 函数,将 flushCallbacks 函数放入异步任务队列。如果异步任务队列中已经存在 flushCallbacks 函数,等待其执行完成以后再放入下一个 flushCallbacks 函数。
flushCallbacks 函数负责执行 callbacks 数组中的所有 flushSchedulerQueue 函数。
flushSchedulerQueue 函数负责刷新 watcher 队列,即执行 queue 数组中每一个 watcher 的 run 方法,从而进入更新阶段,比如执行组件更新函数或者执行用户 watch 的回调函数。
nexttick就是在dom更新后执行延迟回调,可以获取更新后的dom
1.先判断对promise等的支持度,能用微任务就用微任务。宏任务比微任务实效性差。一般把一个宏任务开始到微任务队列清空视为一次循环,两个宏任务之间时长肯定比两个微任务长;settimeout最短延迟为4ms。
2.flushcallback函数,遍历执行回调函数
3.nexttick,里面就是我们写的回调函数。供flushcallback用。将flushcallback放入异步任务里
三、源码
1.Vue,Observer,Compiler,Dep,Watcher
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
 100
 101
 102
 103
 104
 105
 106
 107
 108
 109
 110
 111
 112
 113
 114
 115
 116
 117
 118
 119
 120
 121
 122
 123
 124
 125
 126
 127
 128
 129
 130
 131
 132
 133
 134
 135
 136
 137
 138
 139
 140
 141
 142
 143
 144
 145
 146
 147
 148
 149
 150
 151
 152
 153
 154
 155
 156
 157
 158
 159
 160
 161
 162
 163
 164
 165
 166
 167
 168
 169
 170
 171
 172
 173
 174
 175
 176
 177
 178
 179
 180
 
 | 
 
 
 class Vue {
 constructor(options) {
 
 this.$options = options || {};
 this.$data = options.data || {};
 this.$el = typeof options.el === 'string' ? document.querySelector(options.el) : options.el;
 
 this._proxyData(this.$data)
 
 new Observer(this.$data)
 
 new Compiler(this)
 }
 _proxyData(data) {
 
 Object.keys(data).forEach(key => {
 Object.defineProperty(this, key, {
 
 enumerable: true,
 configurable: true,
 get() {
 return data[key]
 },
 set(newValue) {
 if (newValue === data[key]) {
 return
 } else {
 data[key] = newValue;
 }
 }
 })
 })
 
 }
 }
 class Observer {
 constructor(data) {
 this.walk(data)
 }
 
 
 walk(data) {
 if (!data || typeof data != 'object') {
 return
 }
 Object.keys(data).forEach(key => {
 this.defineReactive(data, key, data[key])
 })
 }
 
 defineReactive(data, key, val) {
 
 let dep = new Dep()
 const that = this
 
 this.walk(val)
 Object.defineProperty(data, key, {
 enumerable: true,
 configurable: true,
 get() {
 
 Dep.target && dep.addSub(Dep.target)
 return val;
 },
 set(newValue) {
 if (val === newValue) {
 return
 }
 
 that.walk(newValue)
 val = newValue;
 
 dep.notify()
 }
 })
 }
 }
 class Compiler {
 }
 
 
 
 
 class Dep {
 
 constructor() {
 this.subs = []
 }
 
 addSub(sub) {
 if (sub && sub.update) {
 this.subs.push(sub)
 }
 }
 
 notify() {
 this.subs.forEach(sub => {
 sub.update()
 })
 }
 }
 
 
 class Watcher {
 constructor(vm, key, cb) {
 this.vm = vm;
 
 this.key = key;
 
 this.cb = cb;
 
 Dep.target = this;
 
 this.oldValue = vm[key]
 Dep.target = null;
 }
 
 update() {
 
 if (this.lazy) {
 
 
 this.dirty = true
 } else if (this.sync) {
 
 
 
 
 this.run()
 } else {
 
 queueWatcher(this)
 }
 }
 }
 
 
 
 export function queueWatcher (watcher: Watcher) {
 const id = watcher.id
 
 if (has[id] == null) {
 
 has[id] = true
 if (!flushing) {
 
 queue.push(watcher)
 } else {
 
 
 
 let i = queue.length - 1
 while (i > index && queue[i].id > watcher.id) {
 i--
 }
 queue.splice(i + 1, 0, watcher)
 }
 
 if (!waiting) {
 waiting = true
 
 if (process.env.NODE_ENV !== 'production' && !config.async) {
 
 
 flushSchedulerQueue()
 return
 }
 
 
 
 
 
 nextTick(flushSchedulerQueue)
 }
 }
 }
 
 | 
2.nexttick
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 
 | const callbacks = []let pending = false
 
 
 
 
 
 
 
 
 
 
 
 
 
 export function nextTick (cb?: Function, ctx?: Object) {
 let _resolve
 
 callbacks.push(() => {
 if (cb) {
 
 try {
 cb.call(ctx)
 } catch (e) {
 handleError(e, ctx, 'nextTick')
 }
 } else if (_resolve) {
 _resolve(ctx)
 }
 })
 if (!pending) {
 pending = true
 
 timerFunc()
 }
 
 if (!cb && typeof Promise !== 'undefined') {
 return new Promise(resolve => {
 _resolve = resolve
 })
 }
 }
 
 
 
 
 
 
 
 
 function flushSchedulerQueue () {
 
 
 
 
 
 
 
 }
 
 | 
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 
 | let timerFunc
 if (typeof Promise !== 'undefined' && isNative(Promise)) {
 const p = Promise.resolve()
 
 timerFunc = () => {
 
 p.then(flushCallbacks)
 
 
 
 
 
 if (isIOS) setTimeout(noop)
 }
 isUsingMicroTask = true
 } else if (!isIE && typeof MutationObserver !== 'undefined' && (
 isNative(MutationObserver) ||
 
 MutationObserver.toString() === '[object MutationObserverConstructor]'
 )) {
 
 
 
 
 let counter = 1
 const observer = new MutationObserver(flushCallbacks)
 const textNode = document.createTextNode(String(counter))
 observer.observe(textNode, {
 characterData: true
 })
 timerFunc = () => {
 counter = (counter + 1) % 2
 textNode.data = String(counter)
 }
 isUsingMicroTask = true
 } else if (typeof setImmediate !== 'undefined' && isNative(setImmediate)) {
 
 timerFunc = () => {
 setImmediate(flushCallbacks)
 }
 } else {
 
 timerFunc = () => {
 setTimeout(flushCallbacks, 0)
 }
 }
 
 | 
参考
https://blog.csdn.net/Guolicheng_/article/details/119463976
https://blog.csdn.net/weixin_43097944/article/details/116134903?spm=1001.2101.3001.6650.9&utm_medium=distribute.pc_relevant.none-task-blog-2%7Edefault%7EBlogCommendFromBaidu%7Edefault-9-116134903-blog-107348545.pc_relevant_aa&depth_1-utm_source=distribute.pc_relevant.none-task-blog-2%7Edefault%7EBlogCommendFromBaidu%7Edefault-9-116134903-blog-107348545.pc_relevant_aa&utm_relevant_index=10