vue的状态更新方式(异步更新解决)

目录

  • 状态更新(异步更新解决)
    • 解决方案
  • 异步更新及nexttick
    • 为什么需要异步更新
    • nextTick 原理

状态更新(异步更新解决) 在vue中状态更新是异步的,这一点和react中的setstate类似。

解决方案
非组件解决方案:
{{message}}

var vm = new Vue({el: '#example',data: {message: '123'}})vm.message = 'new message' // 更改数据vm.$el.textContent === 'new message' // falseVue.nextTick(function () {vm.$el.textContent === 'new message' // true})

在组件内使用 vm.$nextTick() 实例方法特别方便,因为它不需要全局 Vue ,并且回调函数中的 this 将自动绑定到当前的 Vue 实例上:
Vue.component('example', {template: '{{ message }}',data: function () {return {message: '没有更新'}},methods: {updateMessage: function () {this.message = '更新完成'console.log(this.$el.textContent) // => '没有更新'this.$nextTick(function () {console.log(this.$el.textContent) // => '更新完成'})}}})

因为 $nextTick() 返回一个 Promise 对象,所以你可以使用新的 ES2016 async/await 语法完成相同的事情:methods: {
updateMessage: async function () {this.message = 'updated'console.log(this.$el.textContent) // => '未更新'await this.$nextTick()console.log(this.$el.textContent) // => '已更新'}}


异步更新及nexttick
为什么需要异步更新
vue为了避免频繁的操作DOM,采用异步的方式更新DOM。这些异步操作会通过nextTick函数将这些操作以cb的形式放到任务队列中(以微任务优先),当每次tick结束之后就会去执行这些cb,更新DOM。
异步更新内部是最重要的就是nextTick方法,它负责将异步任务加入队列和执行异步任务。VUE 也将它暴露出来提供给用户使用。在数据修改完成后,立即获取相关DOM还没那么快更新,使用nextTick便可以解决这一问题。
【vue的状态更新方式(异步更新解决)】
nextTick 原理
在下次DOM更新循环结束之后执行的延迟回调。在修改数据之后立即使用该方法,获取更新后的DOM。
/*存放异步执行的回调*/const callbacks = [] /*一个标记位,如果已经有timerFunc被推送到任务队列中去则不需要重复推送*/let pending = false/*一个函数指针,指向函数将被推送到任务队列中,等到主线程任务执行完时,任务队列中的timerFunc被调用*/let timerFunc/*推送到队列中下一个tick时执行cb 回调函数ctx 上下文*/export function nextTick (cb?: Function, ctx?: Object) {let _resolve// 第一步 传入的cb会被push进callbacks中存放起来callbacks.push(() => {if (cb) {try {cb.call(ctx)} catch (e) {handleError(e, ctx, 'nextTick')}} else if (_resolve) {_resolve(ctx)}})// 检查上一个异步任务队列(即名为callbacks的任务数组)是否派发和执行完毕了。pending此处相当于一个锁if (!pending) {// 若上一个异步任务队列已经执行完毕,则将pending设定为true(把锁锁上)pending = true// 调用判断Promise,MutationObserver,setTimeout的优先级timerFunc()}// 第三步执行返回的状态if (!cb && typeof Promise !== 'undefined') {return new Promise(resolve => {_resolve = resolve})}}

Vue 在更新 DOM 时是异步执行的。只要侦听到数据变化,Vue 将开启一个队列,并缓冲在同一事件循环中发生的所有数据变更。
如果同一个 watcher 被多次触发,只会被推入到队列中一次。这种在缓冲时去除重复数据对于避免不必要的计算和 DOM 操作是非常重要的。
然后,在下一个的事件循环“tick”中,Vue 刷新队列并执行实际 (已去重的) 工作。
Vue 在内部对异步队列尝试使用原生的 Promise.then、MutationObserver 和 setImmediate,如果执行环境不支持,则会采用 setTimeout(fn, 0) 代替。
以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

    推荐阅读