詳解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 start async1 start async2 promise1 script end promise2 async1 end setTimeout
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 startsetTimeout被放在最后调用执行
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,输出setTimeoutfinish
4、参考
作者:godbmw
链接:https://www.jianshu.com/p/5938d9fdbccd
共同學習,寫下你的評論
評論加載中...
作者其他優質文章