详解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 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.com
Github:godbmw

转载于:https://www.cnblogs.com/geyouneihan/p/9127324.html

Promiseasync/await都是用于处理JavaScript中的异步操作的工具。它们可以帮助我们更简洁地编写异步代码,并使其更易于阅读和维护。 Promise是一种表示异步操作的对象。它可以有三种状态:Pending(进行中)、Fulfilled(已完成)和Rejected(已拒绝)。当一个Promise对象的状态改变时,它会调用相应的回调函数。 我们可以使用Promise来处理异步操作。通过调用Promise的构造函数并传入一个执行器函数,我们可以创建一个新的Promise对象。执行器函数接受两个参数:resolve和reject。当异步操作成功时,我们可以调用resolve函数将Promise的状态设置为Fulfilled,并传递一个值作为结果。当异步操作失败时,我们可以调用reject函数将Promise的状态设置为Rejected,并传递一个错误对象。 我们可以使用Promise的then方法来处理已完成状态的Promise。then方法接受两个回调函数作为参数:第一个回调函数在Promise成功时被调用,第二个回调函数在Promise失败时被调用。 同时,我们可以使用Promise的catch方法来处理已拒绝状态的Promise。catch方法接受一个回调函数作为参数,该回调函数在Promise被拒绝时被调用。 async/await是基于Promise的语法糖,使异步代码更像同步代码。async函数是一个返回Promise对象的函数,其中使用了await关键字来暂停函数的执行,直到Promise解决(Fulfilled)或拒绝(Rejected)。 在使用async/await时,我们可以使用try/catch语句来捕获Promise中的错误。在try块中,我们可以使用await关键字来等待一个Promise的解决或拒绝。如果Promise被解决,则await表达式将返回解决的值;如果Promise被拒绝,则抛出一个错误对象,可以使用catch块捕获该错误。 总结一下,Promise是一种用于处理异步操作的对象,它有三种状态:进行中、已完成和已拒绝。我们可以使用Promise的then和catch方法来处理已完成和已拒绝状态的Promise。而async/await是基于Promise的语法糖,使异步代码更像同步代码。通过使用async函数和await关键字,我们可以以更简洁的方式编写异步代码,并使用try/catch语句来处理错误。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值