Vue.nextTick()笔记

在 Vue 中,当你修改了响应式数据后,无论是使用 Vue.nextTick() 的回调,还是使用 Promise.resolve().then() 的回调,通常都能够拿到更新后的 DOM。

原理是什么?

这涉及到 Vue 的异步更新队列 机制和 JavaScript 的事件循环(Event Loop)

1. Vue 的异步更新队列

Vue 为了提高性能,并避免不必要的 DOM 操作,它并不会在数据发生变化时立即更新 DOM。相反,它会:

  • 将所有响应式数据的变化收集起来。
  • 同一个事件循环的"下一个微任务(microtask)"阶段,批量地执行这些 DOM 更新操作。
  • 这样可以确保在同一个事件循环周期内,无论数据改变了多少次,DOM 只会更新一次,从而减少重绘和回流,提高性能。

2. JavaScript 的事件循环和微任务/宏任务

JavaScript 的事件循环是其异步执行模型的核心。它将任务分为两种:

  • 宏任务(Macrotasks): 例如 setTimeoutsetIntervalsetImmediate (Node.js)、I/O 操作、UI 渲染等。
  • 微任务(Microtasks): 例如 Promise 的回调 (.then().catch().finally())、MutationObserver 的回调、process.nextTick (Node.js) 等。

事件循环的执行顺序大致是:

  1. 执行当前所有同步代码。
  2. 执行所有微任务队列中的任务。
  3. 执行一个宏任务队列中的任务。
  4. 重复步骤 2 和 3。

关键点: 微任务总是在当前宏任务执行完毕后,下一个宏任务开始之前被执行。

3. Vue.nextTick() 的工作原理

Vue.nextTick() 的设计目的就是为了在 Vue 完成了所有 DOM 更新后执行回调。

  • 内部机制: Vue 内部会尝试使用最高效的异步方式来触发 nextTick 的回调。在支持 Promise 的现代浏览器中,它会优先使用 Promise.then() 来创建微任务。如果不支持 Promise(或在某些特殊情况下),它会降级使用 MutationObserver(也是微任务),最后降级到 setTimeout(0)(宏任务)。
  • 保证: 无论内部使用哪种方式,nextTick 都会确保它的回调函数在 Vue 完成了本次所有响应式数据对应的 DOM 更新之后才执行。这意味着当你进入 nextTick 的回调时,DOM 已经是最新的了。

4. Promise.resolve().then() 的工作原理

Promise.resolve().then() 显式地创建了一个微任务,并将你的回调函数放入微任务队列。

  • 执行时机: 当你在修改响应式数据后立即调用 Promise.resolve().then() 时:

    1. 你修改数据,Vue 将 DOM 更新任务放入其内部的微任务队列。
    2. 你调用 Promise.resolve().then(),将你的回调函数放入微任务队列。
    3. 当前同步代码执行完毕。
    4. 事件循环开始处理微任务队列。
    5. Vue 的 DOM 更新微任务通常会先于你手动创建的 Promise.then() 微任务被处理。 这是因为 Vue 的更新队列是内部管理和优化的,它会在当前事件循环的微任务阶段尽早被清空。
    6. 当 Vue 的 DOM 更新完成,并清空了其内部的微任务后,你的 Promise.then() 回调才会得到执行。

结论:为什么两者都能拿到更新后的 DOM?

  • Vue.nextTick() 它是 Vue 官方提供的 API,专门用于等待 DOM 更新。它内部机制确保了回调在 DOM 更新后执行。
  • Promise.resolve().then() 它创建了一个微任务。由于 Vue 的 DOM 更新也是在微任务阶段完成的,并且通常在其他普通微任务之前被处理,所以当你的 Promise.then() 回调执行时,DOM 也已经更新完毕。

简单来说:

Vue 的 DOM 更新和 Promise.then() 都发生在同一个"微任务阶段"。Vue 的更新任务通常会优先或在同一批次中被处理掉,所以当你的 Promise.then() 回调被执行时,DOM 已经是最新的了。

推荐使用哪一个?

尽管 Promise.resolve().then() 在大多数情况下也能达到目的,但强烈推荐使用 Vue.nextTick()

原因:

  1. 语义明确: nextTick 明确表达了你的意图是等待 Vue 的 DOM 更新,代码可读性更好。
  2. 兼容性与健壮性: nextTick 是 Vue 内部机制的一部分,它会处理不同环境(如浏览器兼容性、Node.js 服务端渲染等)下的最佳异步策略,确保在所有情况下都能正确工作。而直接使用 Promise.then() 可能在某些边缘情况下或特定环境中行为不一致(尽管这种情况很少见)。
  3. 官方推荐: 它是 Vue 提供的官方 API,意味着其行为是稳定和有保障的。

所以,当你需要访问更新后的 DOM 时,请始终优先考虑使用 this.$nextTick(() => { /* 访问更新后的 DOM */ }) 或在 setup 语法中使用 nextTick(() => { /* 访问更新后的 DOM */ })

相关推荐
OpenTiny社区5 分钟前
把 SearchBox 塞进项目,搜索转化率怒涨 400%?
前端·vue.js·github
编程猪猪侠35 分钟前
Tailwind CSS 自定义工具类与主题配置指南
前端·css
qhd吴飞38 分钟前
mybatis 差异更新法
java·前端·mybatis
YGY Webgis糕手之路1 小时前
OpenLayers 快速入门(九)Extent 介绍
前端·经验分享·笔记·vue·web
患得患失9491 小时前
【前端】【vueDevTools】使用 vueDevTools 插件并修改默认打开编辑器
前端·编辑器
ReturnTrue8681 小时前
Vue路由状态持久化方案,优雅实现记住表单历史搜索记录!
前端·vue.js
UncleKyrie1 小时前
一个浏览器插件帮你查看Figma设计稿代码图片和转码
前端
遂心_1 小时前
深入解析前后端分离中的 /api 设计:从路由到代理的完整指南
前端·javascript·api
你听得到111 小时前
Flutter - 手搓一个日历组件,集成单日选择、日期范围选择、国际化、农历和节气显示
前端·flutter·架构
风清云淡_A1 小时前
【REACT18.x】CRA+TS+ANTD5.X封装自定义的hooks复用业务功能
前端·react.js