Vue 中 nextTick 的原理详解

1. 为什么需要 nextTick

Vue 采用 异步渲染机制,当响应式数据发生变化时,Vue 并不会立即更新 DOM,而是将这些变化放入一个 队列 中,并在 同一事件循环(Event Loop)中合并相同的修改,最后执行批量更新。这样做的目的是 提升性能,避免不必要的重复渲染

例如:

<template><div>{{ msg }}</div>
</template><script>
export default {data() {return {msg: "Hello"};},mounted() {this.msg = "Vue";console.log(document.querySelector("div").innerText); // 仍然是 "Hello"this.$nextTick(() => {console.log(document.querySelector("div").innerText); // 现在是 "Vue"});}
};
</script>

为什么 console.log 还是 "Hello"?
因为 Vue 在 this.msg = "Vue"不会立即更新 DOM,而是等本轮事件循环结束后再更新。因此,我们需要 nextTick 来确保获取到更新后的 DOM。


2. nextTick 的原理

Vue 的 nextTick 本质上是一个 异步任务调度器,它会在当前 DOM 任务完成后执行回调。其内部原理主要依赖 微任务(Microtask)和 宏任务(Macrotask)

2.1 任务队列

Vue 内部维护了一个 回调队列(callback queue),当 nextTick 被调用时,它会将回调函数 推入队列,然后等待 Vue 进行 DOM 更新后,再依次执行这些回调。

2.2 任务调度策略

nextTick 采用 优雅降级 的策略,在不同环境下选择最佳的异步方法:

  1. Promise(Microtask)(首选,现代浏览器支持)
  2. MutationObserver(Microtask)(比 setTimeout 更快)
  3. setImmediate(Macrotask)(仅 IE 支持)
  4. setTimeout(Macrotask)(最后的兜底方案)

代码实现:

function nextTick(callback) {const p = Promise.resolve();p.then(callback);
}

在 Vue 3 中:

let callbacks = [];
let pending = false;function flushCallbacks() {pending = false;const copies = callbacks.slice(0);callbacks.length = 0;for (let cb of copies) {cb();}
}export function nextTick(cb) {callbacks.push(cb);if (!pending) {pending = true;Promise.resolve().then(flushCallbacks);}
}

流程解析:

  1. 每次调用 nextTick(cb),将 cb 放入 callbacks 队列中。
  2. 只要 pending === false,就启动 微任务(Promise.then)
  3. 微任务执行 flushCallbacks,依次调用 callbacks 队列中的所有回调。

3. nextTick 在 Vue 2 和 Vue 3 的区别

3.1 Vue 2 的 nextTick

在 Vue 2 中,nextTick 主要依赖:

  • Microtask(Promise.then, MutationObserver)
  • Macrotask(setImmediate, setTimeout)
  • 维护了一个 异步任务队列,用于批量执行 nextTick 回调。

3.2 Vue 3 的 nextTick

Vue 3 主要优化:

  • 只使用 Promise 作为微任务(不再使用 MutationObserver)。
  • 更高效的 异步队列处理机制

Vue 3 中的 nextTick

const resolvedPromise = Promise.resolve();
export function nextTick(fn) {return fn ? resolvedPromise.then(fn) : resolvedPromise;
}

优化点

  • 直接使用 Promise.resolve().then(fn),避免了 Vue 2 复杂的回调队列管理。
  • 如果不传入 fn,则返回一个 Promise,支持 await this.$nextTick()

4. nextTick 的使用场景

4.1 在 DOM 更新后执行操作

<template><div ref="box">{{ message }}</div>
</template><script>
export default {data() {return { message: "Hello" };},methods: {updateMessage() {this.message = "Vue";this.$nextTick(() => {console.log(this.$refs.box.innerText); // "Vue"});}}
};
</script>

4.2 在 watch 中等待 DOM 更新

watch(() => state.count, async (newVal) => {await nextTick();console.log(document.querySelector("#counter").innerText); // 确保 DOM 已更新
});

4.3 在 Vue 3 setup 中使用

import { nextTick, ref } from "vue";setup() {const message = ref("Hello");const updateMessage = async () => {message.value = "Vue";await nextTick();console.log(document.querySelector("#msg").innerText);};return { message, updateMessage };
}


5. 总结

  • nextTick 是 Vue 提供的一个 异步任务调度方法,用于在 DOM 更新后执行回调。
  • Vue 采用 异步批量更新 机制,nextTick 可确保 数据变更后获取到最新的 DOM
  • Vue 内部采用 Promise(Microtask)优先,降级到 MutationObserver / setTimeout 作为备用方案。
  • Vue 3 进一步优化了 nextTick,减少了不必要的复杂度,提升了性能。

你可以简单理解为:

Vue 在修改数据后,不会立即更新 DOM,而是 批量合并修改,并在下一次 事件循环(Event Loop)结束时更新 DOMnextTick 让你可以等到 DOM 更新完成后再执行操作。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/22217.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

挖矿病毒实战分析

场景说明 运维人员再设备巡检过程中发现CPU莫名到达百分百&#xff0c;出现异常&#xff0c;请开始你的应急响应排查 cpu百分百&#xff0c;基本就可以确定是中了挖矿病毒了 我们使用命令ps -aux查看进程&#xff0c;或者使用top -c查看进程&#xff0c;排查挖矿程序 使用t…

蓝桥杯好数

样例输入&#xff1a; 24 输出&#xff1a;7 输入&#xff1a;2024 输出&#xff1a; 150 思路&#xff1a;本题朴素方法的时间复杂度是O(n * log10(n)) &#xff0c;不超时。主要考察能否逐位取数&#xff0c;注意细节pi&#xff0c;这样不会改变i,否则会导致循环错误。 #in…

cs*n 网页内容转为html 加入 onenote

csdn上有好用的内容&#xff0c;我们怎么将它们加到 onenote 里吃灰呢。 一、创建 新html create_html.py import sysdef create_html_file(filename):# 检查是否提供了文件名if not filename:print("请提供HTML文件名")return# 创建HTML内容html_content f"…

【后端基础】布隆过滤器原理

文章目录 一、Bloom Filter&#xff08;布隆过滤器&#xff09;概述1. Bloom Filter 的特点2. Bloom Filter 的工作原理 二、示例1. 添加与查询2. 假阳性 三、Bloom Filter 的操作1、假阳性概率2、空间效率3、哈希函数的选择 四、应用 Bloom Filter 是一种非常高效的概率型数据…

【SPIE出版,见刊快速,EI检索稳定,浙江水利水电学院主办】2025年物理学与量子计算国际学术会议(ICPQC 2025)

2025年物理学与量子计算国际学术会议&#xff08;ICPQC 2025&#xff09;将于2025年4月18-20日在中国杭州举行。本次会议旨在汇聚全球的研究人员、学者和业界专家&#xff0c;共同探讨物理学与量子计算领域的最新进展与前沿挑战。随着量子技术的快速发展&#xff0c;其在信息处…

数据库驱动免费下载(Oracle、Mysql、达梦、Postgresql)

数据库驱动找起来好麻烦&#xff0c;我整理到了一起&#xff0c;需要的朋友免费下载&#xff1a;驱动下载 目前收录了Oracle、Mysql、达梦、Postgresql的数据库驱动的多个版本&#xff0c;后续可能会分享更多。

【2025最新版】Chrome谷歌浏览器如何能恢复到之前的旧版本

背景 今天程序突然出了bug&#xff0c;无法自动测试了&#xff0c;显示Chrome版本不匹配&#xff0c;一看&#xff0c;Chrome居然在我已经关闭升级的情况下&#xff0c;又给我升级了&#xff0c;然后就悲剧了&#xff0c;我的代码不能用了。 于是&#xff0c;做了以下几步&…

网络运维学习笔记 017HCIA-Datacom综合实验01

文章目录 综合实验1实验需求总部特性 分支8分支9 配置一、 基本配置&#xff08;IP二层VLAN链路聚合&#xff09;ACC_SWSW-S1SW-S2SW-Ser1SW-CoreSW8SW9DHCPISPGW 二、 单臂路由GW 三、 vlanifSW8SW9 四、 OSPFSW8SW9GW 五、 DHCPDHCPGW 六、 NAT缺省路由GW 七、 HTTPGW 综合实…

出行项目案例

spark和kafka主要通过Scala实现&#xff0c;Hadoop和HBase主要基于java实现。 通过该项目&#xff0c;主要达到以下目的&#xff1a; &#xff08;1&#xff09;通用的数据处理流程&#xff0c;入门大数据领域 &#xff08;2&#xff09;真实体验大数据开发工程师的工作 &a…

2.21力扣-回溯组合

77. 组合 - 力扣&#xff08;LeetCode&#xff09; 一&#xff1a;JAVA class Solution {List<Integer> list new LinkedList<>();List<List<Integer>> ans new LinkedList<>();public List<List<Integer>> combine(int n, int k)…

智能合约的部署

https://blog.csdn.net/qq_40261606/article/details/123249473 编译 点击图中的 “Compile 1_Storage.sol” 存和取一个数的合约&#xff0c;remix自带 pragma solidity >0.8.2 <0.9.0; /*** title Storage* dev Store & retrieve value in a variable* custom:d…

vmvare kali如何配置桥接模式进行上网

注意点:虚拟机可以PING通物理机,但是PING不通其他的网站。经过收集资料,得知由于是校园网连接,所以DHCP只能分配一个授权的IP地址给连接的主机,由于KALI是桥接物理机,物理机已经获得了这个授权的IP,所以导致桥接的虚拟机无法上网。所以不是因为配置的有问题,而是网络的…

了解Python中的SciPy库

么是 SciPy&#xff1f; SciPy&#xff08;发音为“Sigh Pie”&#xff09;是 Scientific Python 的首字母缩写词&#xff0c;它是 Python 的开源库&#xff0c;用于科学和技术计算。它是 Python 编程语言中称为 Numpy 的基本数组处理库的扩展&#xff0c;旨在支持高级科学和工…

python网络安全怎么学 python做网络安全

&#x1f345; 点击文末小卡片 &#xff0c;免费获取网络安全全套资料&#xff0c;资料在手&#xff0c;涨薪更快 众所周知&#xff0c;python是近几年比较火的语言之一&#xff0c;它具有简单易懂、免费开源、可移植、可扩展、丰富的第三方库函数等特点&#xff0c;Java需要大…

Ubuntu下mysql主从复制搭建

本文介绍mysql 8.4主从集群的搭建&#xff0c;从单个机器安装到集群的配置&#xff0c;整体走了一遍&#xff0c;希望对大家有帮助。mysql 8.4和之前的版本命令上有些变化&#xff0c;大家用来参考。 0、环境 ubuntu&#xff1a; 22.04mysql&#xff1a;8.4 1、安装mysql 1…

MAC快速本地部署Deepseek (win也可以)

MAC快速本地部署Deepseek (win也可以) 下载安装ollama 地址: https://ollama.com/ Ollama 是一个开源的大型语言模型&#xff08;LLM&#xff09;本地运行框架&#xff0c;旨在简化大模型的部署和管理流程&#xff0c;使开发者、研究人员及爱好者能够高效地在本地环境中实验和…

Spring Boot框架总结(超级详细)

前言 本篇文章包含Springboot配置文件解释、热部署、自动装配原理源码级剖析、内嵌tomcat源码级剖析、缓存深入、多环境部署等等&#xff0c;如果能耐心看完&#xff0c;想必会有不少收获。 一、Spring Boot基础应用 Spring Boot特征 概念&#xff1a; 约定优于配置&#…

易基因: ChIP-seq+DRIP-seq揭示AMPK通过调控H3K4me3沉积和R-loop形成以维持基因组稳定性和生殖细胞完整性|NAR

原文&#xff1a;ChIP-seqDRIP-seq揭示AMPK通过调控H3K4me3沉积和R-loop形成以维持基因组稳定性和生殖细胞完整性&#xff5c;NAR 大家好&#xff0c;这里是专注表观组学十余年&#xff0c;领跑多组学科研服务的易基因。 在饥饿等能量胁迫条件下&#xff0c;生物体会通过调整…

uniapp h5端和app端 使用 turn.js

前提:添加页后,添加页与当前页会重叠在一起,不知道为什么,没有找到解决办法 1.h5端 <template><view class"container"><view id"flipbook"><view class"page page1">Page 1</view><view class"page pag…

MySQL数据库(3)—— 表操作

目录 一&#xff0c;创建表 1.1 创建表的SQL 1.2 演示 二&#xff0c;查看表 三&#xff0c;修改表 四&#xff0c;删除表 常用的表操作会涉及到两种SWL语句 DDL&#xff08;Data Definition Language&#xff09;数据定义语言&#xff1a;建表、改表、删表等&#xff0…