继续浏览精彩内容
慕课网APP
程序员的梦工厂
打开
继续
感谢您的支持,我会继续努力的
赞赏金额会直接到老师账户
将二维码发送给自己后长按识别
微信支付
支付宝支付

详解promise、async和await的执行顺序

哔哔one
关注TA
已关注
手记 340
粉丝 93
获赞 543

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 66node v10中,正确输出是:

script start
async1 start
async2
promise1
script end
promise2
async1 end
setTimeout

2、知识点

显然,这考察的是js中的事件循环和回调队列。注意以下几点:

  • Promise优先于setTimeout宏任务。所以,setTimeout回调会在最后执行。

  • Promise一旦被定义,就会立即执行。

  • Promiserejectresolve是异步执行的回调。所以,resolve()会被放到回调队列中,在主函数执行完和setTimeout前调用。

  • await执行完后,会让出线程。async标记的函数会返回一个Promise对象

3、 难点

最令人困惑的,就是async1 endpromise2之后输出

在函数async1中,执行promise由于async2async标记的函数,所以默认返回promise对象)会发现resolve(),然后放入回调队列。

接着执行下方的new Promise中的resolve()输出promise2,再回来输出async1 end

其中,async1函数可以写成以下方式(便于理解):

async function async1(){  console.log('async1 start')
  async2().then( _ => {    console.log( 'async1 end ')
  })
}

3、流程

  1. console.log('script start')输出:script start

  2. setTimeout被放在最后调用

  3. 执行async1函数,输出async1 start。然后,进入async2函数,输出async2,并返回Promise对象。回到async1,由于await,让出线程,async2函数返回的Promise放在回调队列

  4. 新new了一个Promise对象,输出promise1。其中的resolve()被放在回调队列。

  5. console.log('script end')输出:script end

  6. 执行回调队列中,async1返回的Promise对象,对象产生的resolve被放入对调队列。这里不输出任何值。

  7. 执行回调队列中,下方Promise显式声明的resolve,输出promise2

  8. 执行回调队列中,由于async1函数返回的promise对象的resolve,输出async1 end

  9. 执行回调队列中,最后的setTimeout,输出setTimeout

  10. finish

4、参考

  1. promise、async和await之执行顺序的那点事

  2. 半年工作经验今日头条和美团面试题面经分享

  3. 关于node和chrome运行结果不一样的详解



作者:godbmw
链接:https://www.jianshu.com/p/5938d9fdbccd

打开App,阅读手记
0人推荐
发表评论
随时随地看视频慕课网APP