vue异步更新机制

异步更新队列

vue中更新dom是异步执行的,只要侦听到数据变化,Vue 将开启一个队列,并缓冲在同一事件循环中发生的所有数据变更。如果同一个 watcher 被多次触发,只会被推入到队列中一次。(先缓冲,缓冲后再将多次触发属性的最后一次作为promise放入队列;而不是一上来就放队列)

关于上面这句话的理解:vue中数据的更新到dom是异步的,数据的异步更新可以理解成一个promise的微任务;并且对同一个属性值进行多次赋值时,只有最后一次赋值会作为一个promise微任务放到更新队列。具体上代码:

场景1结论:dom异步更新,所谓的异步相当于一个promise微任务

// 场景1:

<div id="example">{{message}}</div>
<script>
  var vm = new Vue({
    el: '#example',
    data: {
      message: '123'
    }
  })
  
    // 碰到这句时,把这句理解成一个promise放在promise队列
  vm.message = 'new message' // 更改数据

  Promise.resolve(100).then(value => {
    console.log("如果上面的赋值是相当于promise,那么这句输出结果应该是new message", vm.$el.textContent);       // 第二步输出:new message
  })
  console.log(vm.$el.textContent);  // 第一步输出:123
</script>

场景2结论:赋值改变响应式属性会被当做一个promise,并且执行的顺序也是跟正常promise队列的顺序一样先进先出

//示例1:
var vm = new Vue({
  el: '#example',
  data: {
    message: '123'
  }
})

vm.message = 'new' // 更改数据

Promise.resolve(100).then(value => {
  console.log("测试第一次赋值", vm.$el.textContent);       // 第二步输出:new (因为vm.message = 'new' 先放进promise队列,所以先执行,所以本行输出数据已经改变为new)
})
console.log(vm.$el.textContent);        // 第一步输出:123




// 示例2:
var vm = new Vue({
  el: '#example',
  data: {
    message: '123'
  }
})

Promise.resolve(100).then(value => {
  console.log("测试第一次赋值", vm.$el.textContent);       // 第二步输出:123 (因为这里vm.message = 'new'这个promise在队列的第二位,所以还没执行,所以此时这里的输出还是123)
})

vm.message = 'new' // 更改数据
console.log(vm.$el.textContent);        // 第一步输出:123

场景3结论:场景3说明了异步更新的执行顺序和原理见代码下方总结

// 示例1:
<div id="example">{{message}}</div>
<script>
  var vm = new Vue({
    el: '#example',
    data: {
      message: '123'
    }
  })

  vm.message = 'msg' // 更改数据

  Promise.resolve(100).then(value => {
    console.log("测试第一次赋值", vm.$el.textContent);     // 第二步输出:new
  })

  vm.message = 'new' // 更改数据
  console.log("==", vm.$el.textContent);        // 第一步123

  Promise.resolve(100).then(value => {
    console.log("测试第二次赋值", vm.$el.textContent);     // 第三步输出:new
  })
</script>



// 示例2:
<div id="example">{{message}}</div>
<script>
  var vm = new Vue({
    el: '#example',
    data: {
      message: '123'
    }
  })

  Promise.resolve(1).then(value => {
    console.log("xxx", vm.$el.textContent);     // 第二步输出:123
  })

  vm.message = 'msg' // 更改数据

  Promise.resolve(2).then(value => {
    console.log("测试", vm.$el.textContent);  // 第三步输出:new
  })

  vm.message = 'new' // 更改数据

  Promise.resolve(3).then(value => {
    console.log("fff", vm.$el.textContent);         // 第四步输出:new
  })
  
  console.log(vm.$el.textContent);          // 第一步输出:123
</script>


// 总结:场景3的示例1和2一起说明vue异步更新原理。以场景3的示例2简单说下流程:主流程顺序执行代码碰到promise1,把promise1放到微任务队列第一位,接着主线程继续执行同步代码,碰到vm.message = 'msg',将vm.message = 'msg'作为promise放到任务队列第二位,并将vm.message属性放在缓冲区中,接着主线程继续执行同步代码发现promise2,将promise2放到微任务队列第三位,主线程继续执行同步代码,碰到 vm.message = 'new',发现vm.message这个属性已经在缓冲区了,所以会把'new'这个值也缓存到缓冲区(但不会将vm.message = 'new'作为微任务放到微任务队列的),接着主线程继续执行同步代码,碰到promise3,将promise3放入微任务队列的第四位,接着主线程继续执行同步代码碰到console.log(vm.$el.textContent);  此时直接执行,则第一步输出123;主线程执行栈为空了,此时会把刚才在缓冲区的vm.message属性的值 'new' 赋值给微任务中排在第二位"vm.message = 'msg'",赋值后vm.message就等于'new';然后主线程按照队列法则先进先出开始执行微任务队列的任务,第一位任务promise1输出xxx123,第二位任务"vm.message = 'new'" 渲染更新到div#example的dom节点上,第三位任务promise2输出 测试new(因为上一个任务已经将new更新到dom节点了)。第四位任务promise3输出 fff new。
主要注意的点是vue会把更新数据的异步任务按照主线程的同步代码执行正常放到队列,但是会将属性缓存到缓冲区,如果执行同步代码时再遇见则更新缓冲区的属性值,而不会再遇见时当做任务放入队列中;最后等到同步代码执行完毕后,会把缓冲区最终的值更新到微任务队列中的异步数据更新任务上, 然后开始从头到尾执行微任务队列。
最后圆满结束。
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 201,924评论 5 474
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 84,781评论 2 378
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 148,813评论 0 335
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,264评论 1 272
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,273评论 5 363
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,383评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,800评论 3 393
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,482评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,673评论 1 295
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,497评论 2 318
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,545评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,240评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,802评论 3 304
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,866评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,101评论 1 258
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,673评论 2 348
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,245评论 2 341

推荐阅读更多精彩内容