为什么 `Promise.then` 总比 `setTimeout(..., 0)` 快?微任务的秘密

一、事件循环的两条队列

要理解微任务,我们必须先回到 JavaScript的事件循环(Event Loop)

它处理两种不同类型任务:

  1. 宏任务队列 (Macrotask):存放着独立的、较大的任务。
  2. 微任务队列 (Microtask):存放着需要被紧急执行的、较小的任务。

二、微任务 vs. 宏任务

特性 微任务 (Microtask) 宏任务 (Macrotask)
常见来源 Promise.then/catch/finallyqueueMicrotask()MutationObserverprocess.nextTick (Node.js) setTimeoutsetIntervalrequestAnimationFrame、I/O操作(文件、网络)、UI事件(点击、键盘)
执行时机 在当前任务执行结束后、下一个任务开始前立即执行。 在上一个任务结束后,由事件循环调度,中间可能穿插渲染和微任务。
执行数量 在一次循环中,会全部清空 在一次循环中,只会执行一个
优先级 。可以看作是当前任务的"收尾工作"。 。可以看作是全新的、独立的工作。
比喻 去银行办业务时,填完一张表后,你需要立即去另一个窗口签字确认。这个"签字"就是微任务,必须立刻完成,才能算当前业务(宏任务)告一段落。 你在银行办完了一项业务(宏任务),然后重新取号,等待叫到你的号再去办另一项完全独立的业务(下一个宏任务)。

三、事件循环示例

typescript 复制代码
// 1. 同步代码入栈执行
console.log('Script Start');

// 2. 遇到 setTimeout,将其回调函数注册到 Web API,计时器结束后,回调被放入「宏任务队列」。
setTimeout(() => {
  console.log('Timeout');
}, 0);

// 3. 遇到 Promise.resolve(),其 .then() 的回调被立即放入「微任务队列」。
Promise.resolve().then(() => {
  console.log('Promise');
});

// 4. 同步代码入栈执行
console.log('Script End');

解析:

  1. 同步代码执行完毕 :调用栈变空。输出 Script StartScript End
  2. 检查微任务队列 :发现里面有一个 () => console.log('Promise')
  3. 清空微任务队列 :执行该任务。输出 Promise
  4. 检查微任务队列:现在空了。
  5. (可能进行渲染)
  6. 取一个宏任务 :从宏任务队列中取出 () => console.log('Timeout')
  7. 执行宏任务 :执行该任务。输出 Timeout

流程结束。这完美解释了我们看到的输出顺序。

总结

如果你喜欢本教程,记得点赞+收藏!关注我获取更多JavaScript/TypeScript开发干货

相关推荐
Laravel技术社区9 分钟前
用PHP8实现斗地主游戏,实现三带一,三带二,四带二,顺子,王炸功能(第二集)
前端·游戏·php
m0_738120721 小时前
应急响应——知攻善防Web-3靶机详细教程
服务器·前端·网络·安全·web安全·php
hh随便起个名7 小时前
力扣二叉树的三种遍历
javascript·数据结构·算法·leetcode
我是小路路呀8 小时前
element级联选择器:已选中一个二级节点,随后又点击了一个一级节点(仅浏览,未确认选择),此时下拉框失去焦点并关闭
javascript·vue.js·elementui
程序员爱钓鱼8 小时前
Node.js 编程实战:文件读写操作
前端·后端·node.js
PineappleCoder8 小时前
工程化必备!SVG 雪碧图的最佳实践:ID 引用 + 缓存友好,无需手动算坐标
前端·性能优化
JIngJaneIL9 小时前
基于springboot + vue古城景区管理系统(源码+数据库+文档)
java·开发语言·前端·数据库·vue.js·spring boot·后端
敲敲了个代码9 小时前
隐式类型转换:哈基米 == 猫 ? true :false
开发语言·前端·javascript·学习·面试·web
澄江静如练_9 小时前
列表渲染(v-for)
前端·javascript·vue.js