Promise—关于catch(你真的了解catch的执行顺序吗)

一、问题

今天突然被同事问到一个问题,以下代码怎么输出:

Promise.all([
   new Promise(res => res(0)), 
   new Promise((res, rej) => rej(1))
]).then(v => {
   console.log('then 1 ====== ', v)    
}).catch(e => {
   console.log('catch 1 ====== ', e)
})

Promise.all([
   new Promise(res => res(0)), 
   new Promise(res => res(1)),
   new Promise(res => res(2)), 
   new Promise(res => res(3))
]).then(v => {
   console.log('then 2 ====== ', v)    
}).catch(e => {
   console.log('catch 2 ====== ', e)
})

一眼看完,我果断回答了如下答案:

catch 1 ======  1
then 2 ======  [0, 1, 2, 3]

然而...翻了个大车,实际代码运行结果是:

then 2 ======  [0, 1, 2, 3]
catch 1 ======  1

一直以来都有个误解,以为then和catch的执行是一个二选一的过程,但这段代码颠覆了我的认知,这是为什么?它的执行过程是怎样的?Promise.all()方法的特点是输入的所有promise都完成则完成,有一个拒绝则拒绝,所以显然不是Promise.all()的问题。

那么其实可以将上面的例子简化成如下代码,后面的分析将以如下代码为例

Promise.reject().then(() => {
  console.log('1-1')
}).catch(() => {
  console.log('1-2')
})


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

这段代码的执行结果是:2-1、1-2

二、执行过程拆解

翻了一下MDN:catch

image.png

也就是说,catch是then的语法糖,obj.catch(fn) 内部其实是调用obj.then(undefined, fn)。

根据Promise/A+规范,Promise.prototype.then(arg1, arg2)两个参数都是可选,分为以下两种情况:

  • 如果promise状态是fullfilled,而arg1是undefined,那么arg1就相当于 x => x
  • 如果promise的状态是rejected,而arg2是undefined,那么arg2是一个throw error

Promise/A+规范:https://promisesaplus.com/

结合上面的解释,代码的执行过程为:

1、执行同步代码,解析Promise.reject().then(fn),满足情况二,相当于:

Promise.reject().then(() => {
  console.log('1-1')
}, () => {
  throw Error();
}).catch(() => {
  console.log('1-2')
})

我们知道then方法的返回值也是一个promise,此时假设调用Promise.reject().then(arg1, arg2)返回的promise为p1。

注意此时推入微任务队列的是arg2,也就是抛出错误的回调(Promise状态改变后,then中的回调才会被推入微任务队列)。之后p1又调用了catch,因为此时p1状态还没有被解决(p1处于pending状态),后面的catch设置的回调不会执行,而p1的状态要等到处理微任务队列时才会被解决。

假设微任务队列为一个数组,那么此时微任务队列中应该为:

[() => {
  throw Error()
}]

2、继续执行同步代码,解析Promise.resolve().then(fn),Promise状态为fulfilled,then中的回调fn被正常推入微任务队列。

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

此时微任务队列中应该为:

[() => {
  throw Error()
},() => {
  console.log('2-1')
}]

3、同步代码执行完了,清空微任务队列中的任务:

  1. 取出第一个任务执行,抛出错误,相当于将p1的状态设置为rejected,那么此时后面的catch中的回调函数也被推入微任务队列

此时微任务队列为:

[() => {
  console.log('2-1')
},() => {
  console.log('1-2')
}]
  1. 取出第二个任务执行,打印2-1

此时微任务队列为:

[() => {
  console.log('1-2')
}]
  1. 取出第三个任务执行,打印1-2

此时微任务队列为:微任务队列清空。

[]

这个问题的关键就在于catch内部调用的还是obj.then(undefined, arg2),如果前一个promise的状态是rejected,并且then中第二个参数arg2是undefined,那么arg2是一个throw error,因此状态会顺延到Event Loop下一次Tick。

再看下面的例子:

Promise.reject().then(() => {
  console.log('1-1')
}).catch(() => {
  console.log('1-2')
  throw Error()
}).then(() => {
  console.log('1-3')
}).catch(() => {
  console.log('1-4')
})

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

调用catch方法在没有设置返回值的情况下默认也会返回一个状态为fulfilled的promise,正常情况下会走后面设置的then回调,但由于这里我们在catch中抛出了新的错误,所以又回到了前面出现的问题——后面的then中的arg2为throw error,因此状态再次发生顺延。

输出顺序为:2-1、1-2、1-4

当然,如果要按开头翻车的答案顺序执行,在确定前面的Promise状态为rejected时,不接then,直接调用catch即可。

Promise.reject().catch(() => {
  console.log('1-2')
})

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

输出顺序为:1-2、2-1

三、小结

看到这我想你已经明白大概是怎么回事了,总之,遇到问题稳住不要慌,文档 + 源码阅读能解决百分之八十以上的问题,一定要多思考,多理解,加油打工人~

四、参考

Promise中then和catch的执行过程

MDN:Promise.prototype.catch()

本文由博客一文多发平台 OpenWrite 发布!

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,921评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,635评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,393评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,836评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,833评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,685评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,043评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,694评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,671评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,670评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,779评论 1 332
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,424评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,027评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,984评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,214评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,108评论 2 351
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,517评论 2 343

推荐阅读更多精彩内容