前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >详解promise、async和await的执行顺序

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

作者头像
Javanx
发布2019-09-04 10:23:57
1.7K0
发布2019-09-04 10:23:57
举报
文章被收录于专栏:web秀

前言

对于promise、async和await的执行顺序,很多人都容易弄混,也有很多人只愿意在程序中运用一种,比如我只使用promise,不使用async和await;也有只用async和await,而不是用promise。所以现在来为大家讲讲一起共存需要注意写什么?

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

示例

下面这段promise、async和await代码,请问控制台打印的顺序?

代码语言:javascript
复制
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中,正确输出是:

代码语言:javascript
复制
script start
async1 start
async2
promise1
script end
promise2
async1 end
setTimeout

知识点

显然,这考察的是js中的事件循环和回调队列。注意以下几点: * Promise优先于setTimeout宏任务。所以,setTimeout回调会在最后执行。 * Promise一旦被定义,就会立即执行。 * Promiserejectresolve是异步执行的回调。所以,resolve()会被放到回调队列中,在主函数执行完和setTimeout前调用。 * await执行完后,会让出线程。async标记的函数会返回一个Promise对象

难点

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

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

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

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

代码语言:javascript
复制
async function async1(){
  console.log('async1 start')
  async2().then( _ => {
    console.log( 'async1 end ')
  })
}

流程

代码语言:javascript
复制
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
本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2018年11月5日,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 前言
  • 示例
  • 知识点
  • 难点
  • 流程
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档