Vue2异步更新及nextTick原理

这篇具有很好参考价值的文章主要介绍了Vue2异步更新及nextTick原理。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

vue 官网中是这样描述 nextTick 的

在下次 DOM 更新循环结束之后执行延迟回调。在修改数据之后立即使用这个方法,可以获取更新后的 DOM。

在学习 nextTick 是如何实现之前,我们要先了解下 JavaScript 的执行机制

JavaScript 执行机制

浏览器是多线程的,例如GUI渲染线程、JS引擎线程、事件监听线程等。。。

javascript 执行机制就是借用浏览器的多线程机制,再基于 Event Loop 事件循环机制实现的。其实现了单线程异步效果

Event Loop 步骤大致如下:

  1. 浏览器加载页面时,除了开辟堆栈内存外,还会创建两个队列
    • Web API:任务监听队列,监测异步任务是否可以执行
    • Task Queue:任务队列,分为异步宏任务队列和异步微任务队列
  2. 当主线程自上而下执行代码过程中,如果遇到异步代码,则把异步任务放到 Web API 中去监听
    • 浏览器会开辟新的线程去监听是否可以执行
    • 不会阻碍主线程的渲染,它会继续向下执行同步代码
  3. 当异步任务被监测为可以执行了(有了运行结果),也不会立即去执行,而是在 task queue 中放置一个事件,排队等待执行
    • 根据微任务还是宏任务,放在不同的队列中
    • 谁先进来排队的,谁在各自队伍的最前面
  4. 执行栈中的所有同步任务执行完毕,主线程空闲下来,此时会去 task queue 中把正在排队的事件,按照顺序取出来,进入主线程执行
    • 微任务优先级比较高。当执行栈为空时,先去执行微任务队列中的事件,直到微任务队列为空,才会去执行宏任务队列中的事件
  5. 上述过程会不断重复,也就是常说的事件循环(Event Loop)

task 又分为宏任务(macro task)和微任务(micro task)两大类,在浏览器环境中

  1. 常见的 macro task 有 script(整体代码)、setTimeout/setInterval/setImmediateXMLHttpRequest/fetch,DOM事件(如鼠标点击、滚动页面、放大缩小等),渲染事件(解析 DOM、计算布局、绘制)
  2. 常见的 micro task 有 Promise.then/catch/finallyasync/awaitMutationObserver

需要注意的是!!!如果处理微任务的过程中有新的微任务添加进来了,添加的速度一直比执行快,则永远执行微任务

下面的代码永远不会打印宏任务输出

function macroFn(){
   setTimeout(() => {
     console.log('>>>>MA')
   },0)
}
function microFn(){
    Promise.resolve().then(() => {
        console.log('mi')
        microFn()
    })
}
macroFn()
microFn()

nextTick实现原理

vue2.7 源码中,有一个单独的文件src/core/util/next-tick.js去维护 nextTick,有兴趣的同学可以自行去观看

vue2.7 源码中,nextTick并没有直接使用某个 API ,而是采用了优雅降级的方案去实现异步更新

在内部会尝试使用原生的Promise.then (IE不支持)MutationObserversetImmediate (高版本IE专享),如果执行环境还不支持的话,则会采用 setTimeout(fn, 0)

需要注意的是,我们维护了一个 callbacks,用于存储 nextTick 回调

这样就保证了在同一个 tick 内多次调用 nextTick,只需创建一个异步任务,就可以依次执行 callbacks 中的所有 nextTick 回调。而不是去开启多个异步任务去处理。

let callbacks = [] // 存储 nextTick 回调
let waiting = false // 防抖

// 按照顺序依次执行 callbacks 中的方法
function flushCallbacks() {
  let cbs = callbacks.slice(0)
  waiting = false
  callbacks = []
  cbs.forEach(cb => cb()) 
}


let timerFunc;
if (Promise) {
    timerFunc = () => {
        Promise.resolve().then(flushCallbacks)
    }
}else if(MutationObserver){
    let observer = new MutationObserver(flushCallbacks); // 这里传入的回调是异步执行的
    let textNode = document.createTextNode(1);
    observer.observe(textNode,{
        characterData:true
    });
    timerFunc = () => {
        textNode.textContent = 2;
    }
}else if(setImmediate){
    timerFunc = () => {
       setImmediate(flushCallbacks);
    }
}else{
    timerFunc = () => {
        setTimeout(flushCallbacks);
     }
}

export function nextTick(cb) {
  callbacks.push(cb) // 维护 nextTick 中的 cakllback 方法
  
  if (!waiting) {
    timerFunc()
    waiting = true
  }
}

异步更新

vue 内部的异步更新渲染也使用了 nextTick

在 Watcher 类的 update 更新方法中,我们调用了 queueWatcher 异步队列更新方法,该方法在 vue2.7源码中的 src/core/util/scheduler.js 文件中维护

import { queueWatcher } from './scheduler'

class Watcher {
	...
  // 重新渲染
  update() {
    console.log('watcher-update')
    queueWatcher(this) // watcher 异步更新
  }
}

src/core/util/scheduler.js

import { nextTick } from '../util/next-tick'

/**
 * @name QueueWatcher,内部 watcher 异步更新
 * @decs 把当前的 watcher 暂存起来,在一个tick周期内,不管我们的 update 执行多少次,只会执行一轮刷新操作
 */

let queue = []
let has = {}
let pending = false // 防抖

function flushSchedulerQueue() {
  let flushQueue = queue.slice(0)
  queue = []
  has = {}
  pending = false
  flushQueue.forEach(q => q.run()) // 在刷新的过程中可能还有新的 watcher,重新放到 queue 中
}

// 在一个tick周期内,不管我们的 update 执行多少次,只会执行一轮刷新操作
export function queueWatcher(watcher) {
  const id = watcher.id
  if (!has[id]) {
    queue.push(watcher)
    has[id] = true
    if (!pending) {
      nextTick(flushSchedulerQueue)
      pending = true
    }
  }
}

常见问题

1. nexTick 是异步还是同步?

这个不能一概而论,nextTick 内部既有同步代码又有异步代码。

例如 维护 callbacks 队列是同步任务;执行队列中的方法是异步任务

2. nextTick 回调的执行是微任务还是宏任务?

针对 vue2.7 来说,nextTick并没有直接使用某个 API ,而是采用了优雅降级的方案去实现异步更新。
在内部会尝试使用原生的Promise.then (微任务)MutationObserver (微任务)setImmediate (宏任务),如果执行环境还不支持的话,则会采用 setTimeout (宏任务)

可以理解为 99% 的场景下都是微任务,只有在不支持 Promise 和 MutationObserver API的浏览器中,才会是宏任务,例如 IE9 、IE10

3. 为什么要封装 nextTick?而不是使用某个具体的 API?

优雅降级。尽量使用微任务,尽可能缩短渲染周期

保证统一性。nextTick 可以暴露给用户,保证用户在修改数据之后立即使用这个方法,可以获取更新后的 DOM

this.name = 'libc'

this.$nextTick(()=>{
  console.log(document.querySelector('.user').innerHTML)
});

参考文档

这一次,彻底弄懂 JavaScript 执行机制文章来源地址https://www.toymoban.com/news/detail-408993.html

到了这里,关于Vue2异步更新及nextTick原理的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 揭秘Vue 2中的$nextTick:等待DOM更新的神奇时刻!

    在 Vue 2 中, $nextTick 是一个异步方法,用于在下次 DOM 更新循环结束后执行回调函数。 它的原理可以解析如下: Vue 2 维护了一个队列,用于存储需要延迟执行的回调函数。 下面是一个简单的代码案例,演示了 Vue 2 中的 $nextTick 方法和队列机制: HTML: JavaScript: 在上述代码中,

    2024年02月12日
    浏览(26)
  • Vue中$nextTick主要作用、原理及使用方法

    查看本专栏目录 关于作者 还是大剑师兰特 :曾是美国某知名大学计算机专业研究生,现为航空航海领域高级前端工程师;CSDN知名博主,GIS领域优质创作者,深耕openlayers、leaflet、mapbox、cesium,canvas,webgl,echarts等技术开发,欢迎加底部微信,一起交流。 热门推荐 内容链接

    2024年02月21日
    浏览(30)
  • 【源码&库】Vue3 中的 nextTick 魔法背后的原理

    根据官网的简单介绍, nextTick 是等待下一次 DOM 更新刷新的工具方法。 类型定义如下: 然后再根据官网的详细介绍,我们可以知道 nextTick 的大体实现思路和用法: 当你在 Vue 中更改响应式状态时,最终的 DOM 更新并不是同步生效的,而是由 Vue 将它们缓存在一个队列中,直到

    2024年02月03日
    浏览(31)
  • 【vue2】vue2中的性能优化(持续更新中)

    ⭐ v-for 遍历避免同时使用 v-if ⭐ v-for 中的key绑定唯一的值 ⭐ v-show与v-if对性能的影响 ⭐ 妙用计算属性 ⭐ 使用防抖与节流控制发送频率 ⭐ 路由守卫处理请求避免重复发送请求 ⭐ 使用第三方UI库的引入方式 【前言】 该系列是博主在使用vue2开发项目中常用上的一些小Tips,学

    2024年01月16日
    浏览(31)
  • Vue2相关面试题(持续更新)

    前言 目前这套面试题只适合 初级前端 ,后面会进行深层次补充和拓展以及Vue2源代码的讲解(虽然Vue2今年开始不维护了,但是我的面试题不会止步,冲冲冲)。 在面试的过程中,一定要清楚哪些该说哪些不该说,如果一个知识点不太清楚,就不要做过多的解释,一笔带过就

    2024年02月04日
    浏览(27)
  • 从vue2.0更新到vue3.0

    一、从vue2.0过渡到vue3.0要重新搭建脚手架                 共有两种方式         第一种 :vue-cli : 安装并执行 npm init vue@latest         选择项目功能时: 除了第一项的项目名字外,其他可以暂时No         cd your-project-name         npm install         npm run

    2024年02月03日
    浏览(24)
  • Vue2到3 全套学习内容(持续更新)

    目录 一、Vue 初次上手  1. Vue 概念 2.创建 Vue 实例,初始化渲染  3.插值表达式 4.响应式特性 5.安装 Vue 开发者工具: 装插件调试 Vue 应用 二、Vue 指令  1.内容渲染指令 v-text,v-html  2.1 条件渲染指令 v-show , v-if 2.2 条件渲染指令 v-else, v-else-if 3.事件绑定指令  v-on 1️⃣内

    2024年02月15日
    浏览(27)
  • 【VUE2】VUE2基础知识和原理--超详细--超简介--零基础(一)

    想让Vue工作,就必须创建一个Vue实例,且要传入一个配置对象 demo容器里的代码依然符合html规范,只不过混入了一些特殊的Vue语法 demo容器里的代码被称为【Vue模板】 Vue实例和容器是一一对应的 真实开发中只有一个Vue实例,并且会配合着组件一起使用 {{xxx}}是Vue的语法:插值

    2024年02月16日
    浏览(28)
  • 从Vue2到Vue3【七】——Vue2中响应式原理的实现及其缺陷

    内容 链接 从Vue2到Vue3【零】 Vue3简介 从Vue2到Vue3【一】 Composition API(第一章) 从Vue2到Vue3【二】 Composition API(第二章) 从Vue2到Vue3【三】 Composition API(第三章) 从Vue2到Vue3【四】 Composition API(第四章) 从Vue2到Vue3【五】 新的组件(Fragment、Teleport、Suspense) 从Vue2到Vue3【六

    2024年02月15日
    浏览(32)
  • vue2双向数据绑定基本原理

    vue2的双向数据绑定(又称响应式)原理,是通过数据劫持结合发布订阅模式的方式来实现的,通过 Object.defineProperty() 来劫持各个属性的 setter , getter ,在数据变动时发布消息给订阅者,触发相应的监听回调来渲染视图。也就是说数据和视图同步,数据发生变化,视图跟着变化

    2023年04月10日
    浏览(44)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包