PromiseResolveThenableJobTask 的在Promise中的使用

问题:

最近看到一个有意思的面试题:

js 复制代码
Promise.resolve()
  .then(() => {
    console.log(0);
    return Promise.resolve(4);
  })
  .then((res) => {
    console.log(res);
  });

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

结论:

最终 console 输出顺序: 0, 1, 2, 3, 4, 5

原因解释

重点:Promise的处理 与 微任务队列:

解析步骤拆解:

  1. 脚本同步阶段:
  • 两个 Promise.resolve() 都是已解决的,同步地注册各自的第一个 .then 的 reaction。它们会分别创建"返回的 promise"(记为 A_return、B_return),并把对应的 PromiseFulfillReactionJobTask(执行第一个 then 的回调)按注册顺序推入微任务队列。
    • 初始 微任务队列 (FIFO): [A (第一个链的第1个 then 回调), B (第二个链的第1个 then 回调)]
  1. 执行微任务(按队列):

    1. 运行 A:

      • 执行回调,打印 0
      • 回调返回 Promise.resolve(4) ---> 一个 thenable(JSPromise)
      • 为把 A_return 与该 thenable 关联,V8 创建并入队一个 PromiseResolveThenableJobTask(见 src/objects/promise.tqPromiseResolveThenableJobTask 是 Microtask,携带 promise_to_resolvethenablethen 等字段),用于在稍后调用 thenable.then(resolve, reject)
      • 此时,微任务队列[B, PRJ](PRJ = PromiseResolveThenableJobTask)
    2. 运行 B:

    • 打印 1 ,返回 undefined,导致 B_return 被 fulfill 并把 B_return.then(...)(即后续的 handler2)对应的 reaction 入队。
      • 此时,微任务队列[PRJ, B2]
    1. 运行 PRJ(PromiseResolveThenableJobTask):
    • 它会执行 thenable.then(resolve, reject)。因为 thenable 是已解决的 Promise.resolve(4),调用 then 会把一个将调用 resolve(4) 的 reaction 入队(记为 TJ)。

    • PRJ 本身不直接调用 resolve(4),只执行 then 的调用来把 thenable 的 reaction 安排为微任务。

    • 此时,微任务队列[B2, TJ]

    1. 运行 B2(第二链的第二个 then):
    • 打印 2 ,返回 undefined,注册并入队 B3(下一 then 的 reaction)。

    • 此时,微任务队列[TJ, B3]

    1. 运行 TJ(thenable 的反应,执行 resolve(4)):
    • resolve(4) 满足了 A_return,把 A_return 的后续 reaction(即 A 的第二个 then,也就是打印 res 的回调 A2)入队。

    • 此时,微任务队列[B3, A2]

    1. 运行 B3:
    • 打印 3,入队 B4(最后一个 then 的 reaction)。

    • 此时,微任务队列[A2, B4]

    1. 运行 A2:打印 4
    2. 运行 B4:打印 5

要点总结:

  • then 回调返回一个 thenable(这里是 Promise.resolve(4))时,规范要求通过一个 PromiseResolveThenableJob(微任务)去"取得 then 并调用它",因此会生成一个专门的微任务(PRJ)来调用 then。该 PRJ 会排在当时队列的尾部。
  • thenable.then(resolve, reject) 本身再产生一个微任务(TJ)来真正调用 resolve(4)。因此 A_return 的 fulfill(A2)要等到 TJ 执行后才会被入队并最终运行。
  • 因为多个微任务在运行期间会继续入队(并保持 FIFO),所以第二条 promise 链的中间 then(打印 2、3)会在 A_return 的最终 reaction(打印 4)之前被调度并打印。
相关推荐
鱼毓屿御11 小时前
如何给用户添加权限
前端·javascript·vue.js
JustHappy11 小时前
「web extensions🛠️」有关浏览器扩展,开发前你需要知道一些......
前端·javascript·开源
xixixin_11 小时前
【JavaScript 】从 || 到??:JavaScript 空值处理的最佳实践升级
开发语言·javascript·ecmascript
belldeep11 小时前
python:用 Flask 3 , mistune 2 和 mermaid.min.js 10.9 来实现 Markdown 中 mermaid 图表的渲染
javascript·python·flask
凉辰12 小时前
使用uni.createInnerAudioContext()播放指定音频(踩坑分享功能)
开发语言·javascript·音视频
echoVic12 小时前
多模型支持的架构设计:如何集成 10+ AI 模型
java·javascript
echoVic12 小时前
AI Agent 安全权限设计:blade-code 的 5 种权限模式与三级控制
java·javascript
David凉宸12 小时前
Vue 3 + TS + Vite + Pinia vs Vue 2 + JS + Webpack + Vuex:对比分析
javascript·vue.js·webpack
boooooooom12 小时前
Pinia必学4大核心API:$patch/$reset/$subscribe/$onAction,用法封神!
javascript·vue.js·面试
不会敲代码112 小时前
解密JavaScript内存机制:从执行上下文到闭包的全景解析
javascript