当前位置: 首页 > news >正文

深入理解 JavaScript 事件循环:宏任务与微任务的执行机制

深入理解 JavaScript 事件循环:宏任务与微任务的执行机制

JavaScript 是单线程语言,但它却能处理复杂的并发操作(如网络请求、定时器、用户交互),这背后的秘密武器就是事件循环(Event Loop)。本文将深入拆解宏任务与微任务的执行逻辑,通过代码示例帮你彻底搞懂执行顺序。

TL;DR

  • 核心机制:JS 引擎执行同步代码 -> 清空微任务队列 -> 尝试 DOM 渲染 -> 执行一个宏任务-> 清空微任务队列 -> … 循环往复。
  • 微任务(MicroTask):优先级高,在当前宏任务结束后立即执行。包括Promise.thenprocess.nextTick(Node)、MutationObserver
  • 宏任务(MacroTask):优先级低,每次循环只执行一个。包括setTimeoutsetIntervalsetImmediate(Node)、I/O、UI Rendering。
  • 关键点:微任务队列总是会在下一个宏任务开始之前被清空。

1. 为什么需要事件循环?

JavaScript 的设计初衷是作为浏览器脚本语言,主要用途是与用户互动和操作 DOM。如果它是多线程的,一个线程在删除 DOM 节点,另一个线程在编辑该节点,会带来复杂的同步问题。因此,JS 选择单线程执行。

为了不阻塞主线程(例如等待一个 5秒的 API 请求),JS 引入了异步非阻塞机制,而事件循环正是协调同步代码与异步回调执行顺序的调度员。

2. 宏任务与微任务的分类

并不是所有的异步任务都是一样的。它们被分为两类队列:

微任务 (MicroTask)

通常是由代码本身产生的任务,优先级较高,需要在当前同步代码执行完后立即处理。

  • Promise.then/.catch/.finally
  • process.nextTick(Node.js 环境,优先级高于 Promise)
  • MutationObserver(监听 DOM 变化)
  • queueMicrotaskAPI

宏任务 (MacroTask)

通常是由宿主环境(浏览器或 Node)发起的任务,每次事件循环只取一个执行。

  • setTimeout/setInterval
  • setImmediate(Node.js)
  • requestAnimationFrame(UI 渲染前执行,归类有些特殊,通常视为渲染阶段的一部分)
  • I/O 操作 (文件读写、网络请求回调)
  • UI Rendering (浏览器绘制)
  • <script>(整体代码本身算第一个宏任务)

3. 事件循环的完整流程

标准的 Event Loop 流程如下:

  1. 执行同步代码(这本身属于第一个宏任务)。
  2. 检查微任务队列
    • 如果队列不为空,取出队首任务执行。
    • 执行过程中如果产生了新的微任务,追加到队尾,继续执行直到队列清空
  3. UI 渲染阶段(浏览器视情况决定是否渲染):
    • 检查是否需要更新 UI。
    • 执行requestAnimationFrame回调(如果在渲染前)。
  4. 执行宏任务
    • 从宏任务队列中取出一个任务执行。
    • 执行完后,回到第 2 步(再次清空微任务)。

口诀:同步走完清微任务,渲染之后取宏任务。

4. 实战代码解析

案例一:基础顺序

console.log('1');// 同步setTimeout(()=>{console.log('2');// 宏任务},0);Promise.resolve().then(()=>{console.log('3');// 微任务});console.log('4');// 同步

解析

  1. 执行同步代码:打印'1',打印'4'
  2. 清空微任务:执行 Promise 回调,打印'3'
  3. 执行宏任务:执行 setTimeout 回调,打印'2'
    结果1 -> 4 -> 3 -> 2

案例二:微任务插队与嵌套

console.log('Start');setTimeout(()=>{console.log('Timeout');// 宏任务},0);Promise.resolve().then(()=>{console.log('Promise 1');// 微任务 1// 微任务中产生新的微任务Promise.resolve().then(()=>{console.log('Promise 2');// 微任务 2});});console.log('End');

解析

  1. 同步打印'Start','End'
  2. 检查微任务队列:发现Promise 1,执行并打印。
  3. Promise 1执行时注册了Promise 2,追加到当前微任务队列尾部。
  4. 微任务队列未空,继续执行Promise 2,打印。
  5. 微任务清空完毕,去宏任务队列取Timeout执行。
    结果Start -> End -> Promise 1 -> Promise 2 -> Timeout

案例三:async/await 的本质

async/await只是 Promise 的语法糖。await这一行右边的代码是同步执行的,await下面的代码相当于放在了Promise.then中,属于微任务。

asyncfunctionasync1(){console.log('async1 start');awaitasync2();// 下面这行相当于 .then(() => console.log('async1 end'))console.log('async1 end');}asyncfunctionasync2(){console.log('async2');}console.log('script start');setTimeout(function(){console.log('setTimeout');},0);async1();newPromise(function(resolve){console.log('promise1');// Promise 构造函数内是同步的resolve();}).then(function(){console.log('promise2');});console.log('script end');

深度解析

  1. script start(同步)
  2. setTimeout注册宏任务。
  3. 调用async1:
    • 打印async1 start(同步)。
    • 调用async2,打印async2(同步)。
    • 遇到await,将async1 end放入微任务队列 (微任务1)。
  4. new Promise:
    • 打印promise1(同步)。
    • resolve()触发then,将promise2放入微任务队列 (微任务2)。
  5. 打印script end(同步)。
  6. 同步结束,清空微任务
    • 执行微任务1:打印async1 end
    • 执行微任务2:打印promise2
  7. 微任务空,执行宏任务
    • 打印setTimeout

结果
script start->async1 start->async2->promise1->script end->async1 end->promise2->setTimeout

(注:旧版 Chrome 曾有 Bug 导致 async1 end 比 promise2 慢,但在现代浏览器中已符合标准,遵循入队顺序)

5. 易错点与注意事项

  1. Promise 构造函数是同步的new Promise(fn)中的fn会立即执行,只有.then中的回调才是微任务。
  2. 微任务饿死宏任务:如果你在微任务中无限循环地添加新的微任务(例如递归 Promise),那么主线程会一直被占用,宏任务永远无法执行,页面会卡死(类似while(true))。
  3. UI 渲染时机:通常浏览器会在清空微任务之后、执行下一个宏任务之前尝试渲染。如果微任务执行时间过长,会阻塞渲染导致掉帧。
  4. Node.js 的差异
    • process.nextTick优先级高于 Promise。
    • 早期的 Node (v10及以前) 在执行完一个阶段的所有宏任务后才清空微任务,但 Node v11+ 已修改为与浏览器一致:每执行完一个宏任务就清空一次微任务

总结

掌握事件循环的关键在于分清同步代码微任务宏任务的层级。始终记住:微任务是 VIP 通道,必须优先走完;宏任务是普通通道,一次只能走一个。

http://www.jsqmd.com/news/78755/

相关文章:

  • 基于模型预测控制与滚动时域估计应用于移动机器人研究附Matlab代码
  • JavaScript函数式编程限流实战:从零构建高性能API保护系统
  • 3D模型压缩革命:5分钟掌握Draco核心技术实战指南
  • Figma汉化插件实战:跨国团队协作的救星
  • Claude Code Router多模型路由配置完全指南
  • 收藏必备!Memento框架:让大模型智能体在实践中成长,而非重复训练
  • 基于线性伽马分布回归模型(gamma)的多变量时间序列预测 gamma多变量时间序列 matl...
  • Lodash 源码精读:防抖节流的实现细节与边界场景
  • LightRAG实战手册:3步打造智能检索系统
  • 小白必看:虚拟内存是什么?C盘文件能删吗?
  • 误删Temp文件如何恢复?完整解决方案
  • 储能变流器三相并网电压矢量控制控制(双向充放电) 0.0~0.7s:储能向电网供电50kW 0...
  • 基于模型预测算法的混合储能微电网双层能量管理系统研究附Matlab代码
  • std::string vs C字符串:性能对比实测
  • 【珍藏干货】企业级AI Agent前端操控新范式:从“命令模式“到“原子化指令“的工程实践
  • 免费获取完整88键钢琴音阶:高品质WAV音频资源大全
  • 106-110 操作内联样式,获取元素的样式,其他样式相关的属性
  • 企业级PVE集群部署实战:从单机到高可用
  • UE5 材质-25-各种节点:点乘dot,VertexNormalWS 节点与 CameraVectorWS 节点,
  • 基于COMSOL平台的热流固耦合压缩空气模型:多场耦合的应力场、温度场与渗流场分析
  • AI应用开发工程师完全指南:从Java转AI,学习路线与必备技能,建议收藏!
  • LoRa+Mesh,利尔达烽火组网方案破解覆盖与灵活难题
  • tar命令进阶技巧:比传统操作快3倍的5个方法
  • DIgSILENT-PowerFactory终极指南:从零开始掌握电力系统仿真
  • Linux小白也能懂:fcitx5中文输入法安装使用图解
  • Windows 11离线安装.NET Framework 3.5终极指南
  • libimagequant:图像量化的终极指南
  • 在生产环境中部署KVCache的5个最佳实践
  • 如何快速分析C盘里到底是什么东西占用了最多空间?
  • 告别逆流风险!安科瑞WiFi防逆流表,极简安装,智慧用电