【揭秘Vue】nextTick的神秘面纱:原理与作用一览无余!

前言 - JavaScript执行原理EventLoop

JavaScript执行原理涉及到Event Loop(事件循环)的概念。JavaScript是一种单线程语言,意味着它一次只能执行一个任务。然而,JavaScript经常需要处理异步任务,比如网络请求、定时器等,而这些任务的执行不能阻塞主线程。这时候就需要Event Loop来协调异步任务的执行。

Event Loop的基本工作原理:

  1. 执行栈(Call Stack): 执行栈是存储函数调用的栈结构。当一个函数被调用,它就会被推入执行栈,执行完毕后就会被弹出。这是一个同步执行的过程。
  2. 消息队列(Message Queue): 消息队列用来存放异步任务的回调函数。当异步任务完成后,会将其回调函数放入消息队列
  3. 事件循环(Event Loop): Event Loop是一个持续运行的循环,负责检查执行栈和消息队列。当执行栈为空时,Event Loop会检查消息队列是否有待执行的任务。如果有,就将任务的回调函数推入执行栈执行。这个过程一直循环进行,形成了事件循环。
  4. 宏任务(Macro Task)和微任务(Micro Task): 在事件循环中,任务分为宏任务和微任务。宏任务包括整体的script代码、setTimeout、setInterval等,而微任务包括Promise、process.nextTick等。微任务的优先级高于宏任务,会在当前宏任务执行完毕后立即执行。

整个执行过程可以简要描述为:

  • 执行栈执行同步任务。
  • 当遇到异步任务时,将其回调函数加入消息队列。
  • 当执行栈为空时,Event Loop检查消息队列。
  • 如果消息队列中有任务,就将任务的回调函数推入执行栈执行。
  • 重复以上过程。

下面是一个简单的例子,演示Event Loop的执行过程:

console.log('Start');setTimeout(function() {console.log('Timeout callback');
}, 0);Promise.resolve().then(function() {console.log('Promise callback');
});console.log('End');

在上面的例子中,执行顺序是:

  1. 打印 ‘Start’
  2. 打印 ‘End’
  3. 打印 ‘Promise callback’
  4. 打印 ‘Timeout callback’

这是因为Promise的回调函数属于微任务,它会在当前宏任务执行完毕后立即执行,而setTimeout的回调函数属于宏任务,会在下一个宏任务中执行。

$nextTick的原理

Vue的nextTick其本质是对JavaScript执行原理EventLoop的一种应用。

nextTick的核心是利用了如PromiseMutationObserversetImmediatesetTimeout等原生JavaScript方法来模拟对应的微/宏任务的实现,本质是为了利用JavaScript的这些异步回调任务队列来实现Vue框架中自己的异步回调队列

具体的解释

nextTick 是 Vue.js 中一个重要的异步更新机制,它用于在 DOM 更新之后执行回调函数。Vue.js通过nextTick来确保在数据变化后,DOM 已经更新。

JavaScript 的事件循环(Event Loop)分为宏任务队列(macrotask queue)和微任务队列(microtask queue)。nextTick的实现利用了这两个队列,以确保在 DOM 更新后执行回调。

在 Vue.js 中,当数据发生变化时,Vue 异步执行 DOM 更新。在这个过程中,Vue 会将需要执行的回调函数推入一个队列中,而不是立即执行它们。这就是 nextTick 的核心机制。

在现代浏览器中,Vue 使用 PromiseMutationObserver 来模拟微任务。如果两者都不可用,Vue 会回退到使用 setTimeout 来模拟宏任务。

下面是一个简化的 nextTick 实现,用于说明其基本原理:

let callbacks = [] //用于存储待执行的回调函数。
let pending = false //用于标识是否已经将回调函数推入队列,避免重复推入。function flushCallbacks() { //用于执行队列中的所有回调函数。pending = falseconst copies = callbacks.slice(0)callbacks.length = 0for (let i = 0; i < copies.length; i++) {copies[i]()}
}function nextTick(cb) { //用于将回调函数推入队列,并在适当的时机执行。callbacks.push(cb)if (!pending) {pending = trueif (typeof Promise !== 'undefined') {Promise.resolve().then(flushCallbacks)} else if (typeof MutationObserver !== 'undefined') {const observer = new MutationObserver(flushCallbacks)const textNode = document.createTextNode('1')observer.observe(textNode, { characterData: true })textNode.data = '2'} else if (typeof setImmediate !== 'undefined') {setImmediate(flushCallbacks)} else {setTimeout(flushCallbacks, 0)}}
}

nextTick 在不同环境下使用了不同的异步执行机制,确保了在现代浏览器和旧版本浏览器中都能够正常工作。这样,Vue 能够在数据变化后,等待 DOM 更新完成后执行用户传入的回调函数,从而实现更可靠的异步更新机制。

$nextTick的作用

nextTick不仅是Vue内部的异步队列的调用方法,同时也允许开发者在实际项目中使用这个方法来满足实际应用中对DOM更新数据时机的后续逻辑处理。

nextTick是典型的将底层JavaScript执行原理应用到具体案例中的示例,引入异步更新队列机制的原因:

  • 如果是同步更新,则多次对一个或多个属性赋值,会频繁触发UI/DOM的渲染,可以减少一些无用渲染。
  • 同时由于VirtualDOM的引入,每一次状态发生变化后,状态变化的信号会发送给组件,组件内部使用VirtualDOM进行计算得出需要更新的具体的DOM节点,然后对DOM进行更新操作。每次更新状态后的渲染过程需要更多的计算,而这种无用功也将浪费更多的性能,所以异步渲染变得更加至关重要。

Vue采用了数据驱动视图的思想,但是在一些情况下,仍然需要操作DOM。有时候,可能遇到这样的情况,DOM1的数据发生了变化,而DOM2需要从DOM1中获取数据,那这时就会发现DOM2的视图并没有更新,这时就需要用到了nextTick了。

由于Vue的DOM操作是异步的,所以,在上面的情况中,就要将DOM2获取数据的操作写在$nextTick中。

所以,在以下情况下,会用到nextTick

  • 在数据变化后执行的某个操作,而这个操作需要使用随数据变化而变化的DOM结构的时候,这个操作就需要方法在nextTick()的回调函数中。
<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>Vue nextTick Example</title>
</head>
<body><div id="app"><p>{{ message }}</p><button @click="changeMessage">Change Message</button><div v-if="showMessage">DOM Updated: {{ updatedMessage }}</div>
</div><script src="https://cdn.jsdelivr.net/npm/vue@2"></script>
<script>
new Vue({el: '#app',data: {message: 'Hello, Vue!',showMessage: false,updatedMessage: ''},methods: {changeMessage() {this.message = 'Updated Message';// 使用 nextTick 确保 DOM 已经更新this.$nextTick(() => {this.updatedMessage = this.$el.querySelector('p').textContent;this.showMessage = true;});}}
});
</script></body>
</html>

在这个例子中,当点击按钮改变message数据时,我们使用this.$nextTick来确保在DOM更新之后获取更新后的文本内容,并在页面上显示。

  • 在Vue生命周期中,如果在created()钩子进行DOM操作,也一定要放在nextTick()的回调函数中。因为在created()钩子函数中,页面的DOM还未渲染,这时候也没办法操作DOM,所以,此时如果想要操作DOM,必须将操作的代码放在nextTick()的回调函数中。
<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>Vue nextTick Example</title>
</head>
<body><div id="app"></div><script src="https://cdn.jsdelivr.net/npm/vue@2"></script>
<script>
new Vue({el: '#app',created() {// 在 created 钩子中进行DOM操作,使用 nextTick 确保 DOM 已经渲染this.$nextTick(() => {const element = document.createElement('p');element.textContent = 'DOM Manipulation in created hook';this.$el.appendChild(element);});}
});
</script></body>
</html>

在这个例子中,我们在Vue实例的created钩子中创建一个<p>元素,并使用this.$nextTick确保DOM已经渲染,然后将这个元素添加到页面中。

源码附件:点此下载

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

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

相关文章

研究前沿| Nature:艰难梭菌引发肠道神经源性炎症的新机制

前言 艰难梭菌感染&#xff08;Clostridioides difficile infection&#xff09;是目前发达国家医院和社区内获得性肠道细菌感染腹泻的最主要原因之一。在美国&#xff0c;每年有约50万例病例和导致约29,000例死亡。艰难梭菌&#xff08;C. difficile&#xff09;是一种产生孢子…

sftp 从windows10向linux(centos7)传输文件

前言背景&#xff1a;该示例是需要从windows10向本地linux系统传输一个qt安装文件&#xff0c;不想或者无法安装xftp这些传输工具&#xff0c;直接通过命令传输&#xff1b; 首先保证windows10 ping通linux系统ip&#xff0c;linux ping 通windows10系统&#xff1b; 注意&am…

Linux:动静态库

目录 一、软硬链接 1、软链接 2、硬链接 二、动态库和静态库 编写一个库 ①静态库 使用静态库的方法 ②动态库 使用动态库的方法 库存在的意义 一、软硬链接 软硬链接的本质区别就是&#xff1a;有无独立的inode 软链接有独立的inode&#xff0c;也就意味着软链接是一…

Java(三)(static,代码块,单例设计模式,继承)

目录 static 有无static修饰的成员变量 有无static修饰的成员方法 static的注意事项 代码块 静态代码块 实例代码块 单例设计模式 饿汉式单例写法 懒汉式单例写法 继承 基本概念 注意事项 权限修饰符 单继承 object 方法重写 子类方法中访问其他成员(成员变量…

虹科分享 | PEAK版本升级,看看有没有你关注的新功能?

号外号外&#xff01;近期PEAK进行了重要的版本升级&#xff0c;这次升级带来了许多令人兴奋的功能优化&#xff0c;助力您的工作流程更加便捷高效。为了帮助您更好地了解PEAK新版本&#xff0c;我们提供了详细的说明和指导&#xff0c;快来看看有没有你关注的新功能&#xff1…

【2023春李宏毅机器学习】快速了解机器学习基本原理

文章目录 机器学习约等于机器自动找一个函数 机器学习分类 regression&#xff1a;输出为连续值classification&#xff1a;输出为一个类别structured learning&#xff1a;又叫生成式学习generative learning 生成有结构的物件&#xff08;如&#xff1a;影像、句子&#xf…

【每日一题】53. 最大子数组和-2023.11.20

题目&#xff1a; 53. 最大子数组和 给你一个整数数组 nums &#xff0c;请你找出一个具有最大和的连续子数组&#xff08;子数组最少包含一个元素&#xff09;&#xff0c;返回其最大和。 子数组 是数组中的一个连续部分。 示例 1&#xff1a; 输入&#xff1a;nums [-2…

色彩的基础知识——适用于camera tuning

#灵感#AWB CC是图像调优中的一个重要色彩部分&#xff0c;了解一些色彩知识。 目录 1、光谱的颜色-------滤光片、颜色风格 2、人眼的明暗视觉-----------侧向光学&#xff0c;对比度、色噪 3、视觉残留-------帧率 4、颜色的明度、色调、饱和度 5、三基色 RGB---------调…

CRM系统怎样帮助企业提高客户服务能力?

在企业的经营过程中&#xff0c;绝不是追求“一锤子买卖”&#xff0c;而是与客户建立长期稳定的关系&#xff0c;从而获得更多的收益。CRM客户关系管理系统顾名思义&#xff0c;就是帮助企业管理客户关系的软件。下面说说&#xff0c;如何通过CRM提高客户服务能力? CRM可以帮…

融合语言模型中的拓扑上下文和逻辑规则实现知识图谱补全11.18

融合语言模型中的拓扑上下文和逻辑规则实现知识图谱补全 摘要1 引言2 相关工作2.1 事实嵌入法2.2 拓扑嵌入方法2.3 规则融合方法2.4 基于LM的方法 3 准备3.1 知识图谱和拓扑上下文3.2 KG中的逻辑规则4.3 三元组嵌入 5 实验和结果5.1 数据集和评价指标 摘要 知识图补全&#xf…

Vue图片URL转File实践[已解决跨域问题]

async imgToFile(url) {const self thisconst image new Image()//let url "https://shenjianblog.oss-cn-shanghai.aliyuncs.com/pic/20220612/sfxs.jpg"// 使用 fetch 获取图像作为 Blob 对象const response await fetch(url.replace("https://shenjianbl…

【算法心得】minus instead of add

https://leetcode.com/problems/minimum-amount-of-time-to-collect-garbage/description/?envType=daily-question&envId=2023-11-20 Here is my code: function garbageCollection(garbage: string[], travel: number[]): number

Django与Ajax

一、什么是Ajax AJAX&#xff08;Asynchronous Javascript And XML&#xff09;翻译成中文就是“异步Javascript和XML”。即使用Javascript语言与服务器进行异步交互&#xff0c;传输的数据为XML&#xff08;当然&#xff0c;传输的数据不只是XML&#xff0c;现在更多使用json…

米贸搜|Facebook精准增粉策略

优化公共主页 1-你的Facebook公共主页是用户对品牌的第一印象&#xff0c;因此&#xff0c;优化主页至关重要。你需要确保主页名称、封面照片、简介、联系方式等信息准确无误&#xff0c;同时&#xff0c;通过设定的一些细节和功能&#xff0c;来提升用户体验和互动性。 发帖…

京东大数据(京东数据采集):2023年Q3线上投影仪品类销售数据分析报告

11月初&#xff0c;某知名投影仪企业发布了2023年三季度财报。数据显示&#xff0c;今年第三季度&#xff0c;公司营收依然不客观&#xff0c;连续第五个季度业绩持续下滑。 从鲸参谋数据也可以看出&#xff0c;今年Q3&#xff0c;京东平台上该品牌的销量环比下滑约35%&#x…

谷歌浏览器版本下载

Chrome 已是最新版本 版本 119.0.6045.160&#xff08;正式版本&#xff09; &#xff08;64 位&#xff09; 自定义chrome https://www.sysgeek.cn/chrome-new-tab-page-customize/ chrome怎么把标签放主页 https://g.pconline.com.cn/x/1615/16153935.html 谷歌浏览器怎么设…

工程项目立项需要做哪些准备?

工程项目立项是一个复杂的过程&#xff0c;需要进行多方面的准备工作。这些准备工作对于项目的顺利进行至关重要&#xff0c;下面将详细介绍工程项目立项需要做哪些准备。 一、项目前期调研 在进行工程项目立项之前&#xff0c;需要进行充分的前期调研。这个阶段的主要目的是了…

进程保活-账号同步实现

账户同步的作用 : 如果应用的数据发生了改变 , 可以通过账户进行同步 , 进而与服务器进行数据同步操作 , 执行同步时 , 系统会拉活对应的应用进程 ; 实现的话&#xff0c;主要是应用 APP 中可以注册 " 账户服务 Service " , 应用安装后 , 如果系统发现应用中有该类型…

MATLAB常用绘图函数的使用

文章目录 绘制一图一线绘制一图多线用法一&#xff1a;plot用法二&#xff1a;hold on 绘制一图多图其他形式的坐标图分段函数绘制方法一&#xff1a;分段写函数的定义域值域方法二&#xff1a;判断定义域方法三&#xff1a;if else 判断 横纵坐标范围设置标题、轴标签、图例、…