深入理解 JavaScript 事件循环机制:单线程中的异步处理核心

news2024/10/6 2:53:31

深入理解 JavaScript 事件循环机制:单线程中的异步处理核心

JavaScript 是一门单线程的编程语言,也就是说它在同一时间只能执行一个任务。然而,现代 Web 应用经常需要处理大量的异步操作,如用户输入、网络请求、定时器等。为了确保在这些操作期间应用的流畅运行,JavaScript 引入了事件循环机制(Event Loop),它使得单线程也能高效地处理异步任务。

本文将深入分析 JavaScript 的事件循环机制及其核心组件,帮助你更好地理解和使用这一强大的异步处理工具。

事件循环机制的核心组件

1. 执行栈(Call Stack)

执行栈是一个 LIFO(后进先出)结构,用来管理所有的同步任务。当函数被调用时,它会被推入执行栈顶,函数执行完毕后才会从栈中弹出。JavaScript 在单线程中执行代码的顺序是严格按照执行栈来完成的。

关键点:由于 JavaScript 是单线程的,执行栈中的同步任务会阻塞其他任务的执行。因此,当执行栈上有耗时的任务时,会导致 UI 渲染、用户输入等操作的延迟。为了解决这个问题,JavaScript 借助事件循环机制来处理异步任务。

2. 消息队列(Message Queue)

消息队列是一个 FIFO(先进先出)结构,用于存放待处理的异步任务。这些任务通常包括宏任务(Macro Task),例如 setTimeoutsetInterval、网络请求的回调等。

任务调度:当执行栈中的所有同步代码执行完毕后,事件循环会从消息队列中取出任务,按顺序将它们放入执行栈中执行。消息队列的存在保证了异步任务不会阻塞同步任务。

3. 微任务队列(Microtask Queue)

微任务队列存储优先级比宏任务更高的 轻量级异步任务 ,通常用于处理一些短小、紧急的任务。微任务队列中的任务包括 Promise 的回调、MutationObserverprocess.nextTick(Node.js)。

优先级:每个宏任务执行完毕后,事件循环会立即处理微任务队列中的所有任务。在处理完微任务队列中的任务之前,事件循环不会继续执行下一个宏任务。

4. Web APIs 和 Node.js APIs

虽然 JavaScript 是单线程的,但浏览器和 Node.js 提供的底层 Web APIs 或 Node.js 系统 APIs(如定时器、网络请求等)可以借助多线程机制处理异步任务。当这些任务完成时,它们的回调函数会被推入消息队列等待执行。

事件循环的执行流程

JavaScript 的事件循环遵循一个简单但高效的流程:

  1. 执行同步代码:事件循环首先会执行执行栈中的同步任务。同步任务依次入栈、执行、出栈,直到栈为空。

  2. 处理微任务:执行栈清空后,事件循环会优先处理微任务队列中的任务。如果微任务在执行过程中产生了新的微任务,这些任务也会立即被执行,直到微任务队列为空。

  3. 处理宏任务:当微任务队列清空后,事件循环会从消息队列中取出 一个宏任务 ,将其放入执行栈中执行。宏任务执行完毕后,事件循环再次处理微任务队列。

  4. 重复循环:事件循环会不断重复上述步骤,保证异步任务与同步任务的协调执行。

宏任务与微任务

宏任务(Macro Task)

宏任务是相对较大的异步任务,每个事件循环中只能执行一个宏任务。常见的宏任务包括:

  • setTimeoutsetInterval:用于设置定时器,回调函数会在指定时间后被推入消息队列。
  • I/O 操作:如文件读取、网络请求等任务的回调。
  • 事件处理器:例如 clickkeydown 等事件的回调函数。
  • UI 渲染任务:浏览器中的重排(Reflow)和重绘(Repaint)。
  • setImmediate(Node.js 环境中): 当前事件循环结束后立即执行的回调。
  • requestAnimationFrame:用于在浏览器中下一帧渲染之前执行的回调。

微任务(Microtask)

微任务优先级高于宏任务,在每次宏任务执行结束后会优先处理。常见的微任务包括:

  • Promise.then, catch, finally:Promise 的回调总是在当前事件循环的微任务队列中调度执行。
  • MutationObserver:DOM 发生变化时的回调。
  • process.nextTick(Node.js):一种特殊的微任务,优先级甚至高于 Promise
  • queueMicrotask:显式将回调函数加入微任务队列。

宏任务与微任务的执行顺序示例

通过以下代码示例,我们可以理解宏任务与微任务的执行顺序:

console.log('Start');

setTimeout(() => {
  console.log('Timeout 1');
}, 0);

Promise.resolve().then(() => {
  console.log('Promise 1');
}).then(() => {
  console.log('Promise 2');
});

console.log('End');

执行过程

  1. console.log('Start')console.log('End') 是同步任务,立即执行。
  2. setTimeout 的回调函数被推入消息队列,等待宏任务调度。
  3. Promise.resolve() 生成的 .then() 回调函数被推入微任务队列。
  4. 同步任务执行完毕后,事件循环会先处理微任务队列,依次输出 Promise 1Promise 2
  5. 最后,事件循环会从消息队列中取出 setTimeout 的回调,输出 Timeout 1

最终输出顺序为:

Start
End
Promise 1
Promise 2
Timeout 1

在这里插入图片描述

宏任务与微任务的列表总结

宏任务:

  • setTimeout
  • setInterval
  • setImmediate(Node.js)
  • requestAnimationFrame
  • I/O 操作
  • 事件处理器(如 clickkeydown 等)
  • postMessage
  • MessageChannel
  • UI 渲染任务(如重排和重绘)

微任务:

  • Promise.then, catch, finally
  • MutationObserver
  • process.nextTick(Node.js)
  • queueMicrotask
  • Async/Await

实际应用场景

1. 异步操作的处理

事件循环机制在处理异步操作时显得尤为重要。无论是网络请求、用户交互,还是定时器的执行,它们的回调函数都不会立即执行,而是通过事件循环的调度机制有序执行。这使得主线程不会因等待异步任务的完成而阻塞。

2. 性能优化

开发者可以利用微任务的优先级特性来优化代码的执行顺序。通过 PromisequeueMicrotask,可以将需要优先处理的任务放入微任务队列,确保它们在当前事件循环中尽快执行。

3. 避免阻塞主线程

事件循环机制通过将耗时任务交由 Web APIs 或 Node.js APIs 处理,避免了同步任务阻塞主线程。这在处理大量用户交互或后台数据处理时至关重要。

总结

JavaScript 的事件循环机制使得单线程环境下也能高效处理异步任务。通过执行栈、消息队列、微任务队列的协调工作,JavaScript 在不阻塞主线程的情况下完成各种异步操作。理解事件循环的工作原理,有助于开发者编写出更加高效、响应迅速的 Web 应用。

掌握宏任务和微任务的优先级以及事件循环的调度逻辑,是优化异步操作和改善用户体验的关键。

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

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

相关文章

Vue的基本用法及模板语法

Vue.js使用了基于 HTML 的模板语法,允许开发者声明式地将 DOM 绑定至底层 Vue实例的数据。所有 Vue.js的模板都是合法的 HTML,所以能被遵循规范的浏览器和 HTML 解析器解析。 在底层的实现上,Vue将模板编译成虚拟 DOM 渲染函数。结合响应系…

实现Xshell与虚拟机中Linux服务器的连接(附常见错误解决)

前言 Xshell是一个强大的安全终端模拟软件,它支持SSH1, SSH2, 以及Microsoft Windows 平台的TELNET 协议。Xshell 通过互联网到远程主机的安全连接以及它创新性的设计和特色帮助用户在复杂的网络环境中享受他们的工作。 本文将介绍Xshell与虚拟机中Linux服务器连接…

前缀线性基——关于目前的理解以及一些样题

怎么说呢?在前几天我总结了了有关线性基的一篇博客,线性基用来去求整个区间的异或最值问题 前缀线性基——用于统计一个区间内的异或最值问题 那么我们如何去统计呢?那么就要去存储一个区间的异或空间线性基,因此我们的思路就是用…

【python】追加写入excel

输出文件运行前(有两张表,“表1”和“Sheet1”): 目录 一:写入单表(删除所有旧工作表,写入新表)二:写入多表(删除所有旧工作表,写入新表&#x…

平衡二叉搜索树之 AVL 树的模拟实现【C++】

文章目录 AVL树的简单介绍全部的实现代码放在了文章末尾准备工作包含头文件类的成员变量 构造函数和拷贝构造swap和赋值运算符重载析构函数findinsert[重要]当parent的平衡因子为1/-1时,如何向上更新祖先节点的平衡因子呢?怎么旋转?左单旋右单…

Windows Ubuntu下搭建深度学习Pytorch训练框架与转换环境TensorRT

Windows Ubuntu下搭建深度学习Pytorch训练框架与转换环境TensorRT JetBrains2024(IntelliJ IDEA、PhpStorm、RubyMine、Rider……)安装包Anaconda Miniconda安装.condarc 文件配置镜像源查看conda的配置和源(channel)自定义conda虚拟环境路径conda常用命…

Chromium 中JavaScript Screen API接口c++代码实现

Screen - Web API | MDN (mozilla.org) Screen Screen 接口表示一个屏幕窗口,往往指的是当前正在被渲染的 window 对象,可以使用 window.screen 获取它。 请注意:由浏览器决定提供屏幕对象,此对象一般通过当前浏览器窗口活动状…

《python语言程序设计》2018版第8章19题几何Rectangle2D类(下)-头疼的几何和数学

希望这个下集里能有完整的代码 一、containsPoint实现 先从网上找一下Statement expected, found Py:DEDENTTAB还是空格呢??小小总结如何拆分矩形的四个点呢.我们来小小的测试一下这个函数结果出在哪里呢???修改完成variable in function should be lowercase 函数变量应该…

No.2 笔记 | 网络安全攻防:PC、CS工具与移动应用分析

引言 在当今数字化时代,网络安全已成为每个人都应该关注的重要话题。本文将总结一次关于网络安全攻防技术的学习内容,涵盖PC端和移动端的恶意程序利用,以及强大的渗透测试工具Cobalt Strike的使用。通过学习这些内容,我们不仅能够了解攻击者的手法,更能提高自身的安全意识和防…

【牛顿迭代法求极小值】

牛顿迭代法求极小值 仅供参考 作业内容与要求 作业内容 作业要求 递交报告 代码 编程实现 计算偏导数 故上述非线性方程组的根可能为 f ( x , y ) f(x, y) f(x,y)的极值点,至于是极小值点还是极大值点或鞍点,就需要使用微积分中的黑塞矩阵来判断了。…

网络基础 【HTTPS】

💓博主CSDN主页:麻辣韭菜💓   ⏩专栏分类:Linux初窥门径⏪   🚚代码仓库:Linux代码练习🚚 💻操作环境: CentOS 7.6 华为云远程服务器 🌹关注我🫵带你学习更多Linux知识…

Linux之实战命令26:timeout应用实例(六十)

简介: CSDN博客专家、《Android系统多媒体进阶实战》一书作者 新书发布:《Android系统多媒体进阶实战》🚀 优质专栏: Audio工程师进阶系列【原创干货持续更新中……】🚀 优质专栏: 多媒体系统工程师系列【…

安卓手机密码忘了怎么办?(只做科普)

注意:只做科普,拒绝利用技术做一些非法事情 科普人:网络安全工程师~DL 科普平台:快手,CSDN,微信公众号,小红书,百度,360 本期文章耗时比较大,如果喜欢&…

数学题-分糖果-答案解析

PDF文档回复:20241005 1[题目描述] 幼儿园有7个小朋友,你是其中之一,有一天你发现无穷多颗糖,最少可以拿16个,最多可以拿23个,你打算拿一些分给小朋友们,分配原则是如果每人(包括你)都可以拿1块糖&#xf…

快速上手C语言【上】(非常详细!!!)

目录 1. 基本数据类型 2. 变量 2.1 定义格式 和 命名规范 2.2 格式化输入和输出(scanf 和 printf) ​编辑 2.3 作用域和生命周期 3. 常量 4. 字符串转义字符注释 5. 操作符 5.1 双目操作符 5.1.1 算数操作符 5.1.2 移位操作符 5.1.3 位操作符…

IDEA下“File is read-only”可能原因及“找不到或无法加载主类”问题的解决

1.File is read-only”可能原因 写代码时想要修改这个静态变量的值,把这个语句注释掉,发现在这个文件中File is read-only无法编辑修改,于是想去掉这个状态 网上查看的解释大多是在File栏目或File->File Properties下可以找到Make File W…

Git介绍--github/gitee/gitlab使用

一、Git的介绍 1.1、学习Git的原因:资源管理 1.2、SCM软件的介绍 软件配置管理(SCM)是指通过执行版本控制、变更控制的规程,以及使用合适的配置管理软件来保证所有配置项的完整性和可跟踪性。配置管理是对工作成果的一种有效保护。 二、版本控制软件 …

常见的基础系统

权限管理系统支付系统搜索系统报表系统API网关系统待定。。。 Java 优质开源系统设计项目 来源:Java 优质开源系统设计项目 | JavaGuide 备注:github和gitee上可以搜索到相关项目

企业必备:搭建大模型应用平台实操教程

最近AI智能体很火,AI智能体平台化产品肯定属于大公司的。但在一些场景下,尤其是对业务数据要求很高的公司,那就只能用私有大模型。不一定完全是为了对外提供服务,对内改造工作流也是需要的。所以 我感觉未来大部分企业都会搞一个…

软考系统分析师知识点二:经济管理

前言 今年报考了11月份的软考高级:系统分析师。 考试时间为:11月9日。 倒计时:35天。 目标:优先应试,其次学习,再次实践。 复习计划第一阶段:扫平基础知识点,仅抽取有用信息&am…