说明:
本文摘自详解promise、async和await的执行顺序。
1、题目和答案
一道题题目:下面这段promise、async和await代码,请问控制台打印的顺序?
async function async1(){console.log('async1 start')await async2()console.log('async1 end')}async function async2(){console.log('async2')}console.log('script start')setTimeout(function(){console.log('setTimeout') },0) async1();new Promise(function(resolve){console.log('promise1')resolve();}).then(function(){console.log('promise2')})console.log('script end')复制代码
上述,在Chrome 66
和node v10
中,正确输出是:
script startasync1 startasync2promise1script endpromise2async1 endsetTimeout复制代码
2、知识点
显然,这考察的是js中的事件循环和回调队列。注意以下几点:
Promise
优先于setTimeout
宏任务。所以,setTimeout
回调会在最后执行。Promise
一旦被定义,就会立即执行。Promise
的reject
和resolve
是异步执行的回调。所以,resolve()
会被放到回调队列中,在主函数执行完和setTimeout
前调用。await
执行完后,会让出线程。async
标记的函数会返回一个Promise
对象
3、 难点
最令人困惑的,就是
async1 end
在promise2
之后输出
在函数async1
中,执行promise
(
由于async2
是async
标记的函数,所以默认返回promise
对象)会发现resolve()
,然后放入回调队列。
接着执行下方的new Promise
中的resolve()
输出promise2
,再回来输出async1 end
。
其中,async1
函数可以写成以下方式(便于理解):
async function async1(){console.log('async1 start')async2().then( _ => {console.log( 'async1 end ')})}复制代码
3、流程
console.log('script start')
输出:script start
setTimeout
被放在最后调用执行async1
函数,输出async1 start
。然后,进入async2
函数,输出async2
,并返回Promise
对象。回到async1
,由于await
,让出线程,async2
函数返回的Promise
放在回调队列。新new了一个Promise
对象,输出promise1
。其中的resolve()
被放在回调队列。console.log('script end')
输出:script end
执行回调队列中,async1
返回的Promise
对象,对象产生的resolve
被放入对调队列。这里不输出任何值。执行回调队列中,下方Promise
显式声明的resolve
,输出promise2
。执行回调队列中,由于async1
函数返回的promise
对象的resolve
,输出async1 end
。执行回调队列中,最后的setTimeout
,输出setTimeout
finish