请教一个 Promise 的问题

2021-01-30 15:18:18 +08:00
 rodrick

分别有下面两个写法,区别就是后者用的 .catch ,为啥同用调用后打印结果不一样呢,前者打印 1,后者打印 2

function myPromiseRace(arr) {
  return new Promise((resolve, reject) => {
    for (let item of arr) {
      item.then(
          res => { resolve(res) },
          err => { reject(err) }
        )
    }
  })
}
function myPromiseRace(arr) {
  return new Promise((resolve, reject) => {
    for (let item of arr) {
      item
        .then(
          res => { resolve(res) }
        )
        .catch(err => {
          reject(err)
        })
    }
  })
}

调用

let p1 = Promise.reject(1),
  p2 = Promise.resolve(2),
  p3 = Promise.resolve(3)

myPromiseRace([p1, p2, p3]).then(res => {
  console.log(res)
}).catch(err => {
  console.log(err);
})
2654 次点击
所在节点    JavaScript
9 条回复
h503mc
2021-01-30 15:34:59 +08:00
第二种写法需要 then 里的函数 reject 掉才会触发下一个的 catch
genal
2021-01-30 16:11:50 +08:00
第二个 mypromise 多了个 catch,相当于多套了一层,所以第一个 promise 更慢,由于 promise 只能 resolve,reject 一次,谁快先输出谁
genal
2021-01-30 16:12:32 +08:00
后面的 resolve,rejecrt 是什么不重要了,promise 只能 resolve,reject 一次
faceRollingKB
2021-01-30 16:13:27 +08:00
从 micro task 的角度看,promise 本体在 current task 中执行,then 里面的回调被添加到 micro task 队列中,而第二种写法里面的 catch 是在 then 所属的 micro task 被执行时才被添加到 micro task 队列的,所以:
第一种写法:三个 promise 依次被添加 then 里面定义的 micro task,得到队列 p1-then 、p2-then 、p3-then,p1-then 的 reject 回调会赢得 race ;
第二种写法:三个 promise 依次被添加 then 里面定义的 micro task,micro task 的执行结果会得到新的 promise,继而依次添加 catch 里面的 micro task,完全执行下去会有 6 个 micro task,依次是 p1-then 、p2-then 、p3-then 、p1-catch 、p2-catch 、p3-catch,p1 的 then 虽然排在第 1 位,但只是返回了一个新的 promise 并将 p1-catch 添加到 micro task 队列,并没有回调 then 的第一个参数,所以 p2-then 赢得 race 。
rodrick
2021-01-30 16:16:01 +08:00
@h503mc 是指 p1 还没 reject 的时候 p2 已经 resolve 了么
@genal 大概明白了
rodrick
2021-01-30 16:18:41 +08:00
@faceRollingKB 啊我明白了!确实是在微任务队列里是有先后执行顺序的,我给这茬忘了,非常感谢
TommyDx
2021-01-30 16:24:40 +08:00
第二种写法,在 p1 的 reject 还没被 catch 的时候,p2 的 resolve 已经被 new Promise 的 resolve 函数输出了
cxe2v
2021-01-30 17:29:39 +08:00
4 楼说的很详细了,我说下简化版本的

第一种写法,是同时将 resolve 和 reject 添加到了 微任务 队列
第二种写法,只是将 resolve 先添加到了微任务队列,然后再依次添加了 catch 到微任务

那么第二种写法在执行的时候,肯定是谁先 resolve 谁就赢了,因为 reject 的那个 catch 微任务是排在所有 resolve 的后面的
Sparetire
2021-01-30 17:30:11 +08:00
第一种写法一个 tick,第二种写法两个 tick

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://www.v2ex.com/t/749836

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX