一道面试题
setTimeout(() => {
console.log('setTimeout');
}, 0)
async function async1() {
console.log('async1 start');
await async2();
console.log('async1 end');
}
async function async2() {
console.log('async2');
}
console.log('script start');
async1();
new Promise((resolve) => {
console.log('promise')
resolve()
}).then(() => {
console.log('promise.then')
})
console.log('script end');
输出结果:
为什么是这个结果呢?
首先定义了一个计时器:
它会被交给浏览器的定时器模块处理,当某个定时器到了可执行状态就会把该定时器扔到消息队列里去,最后执行。
然后定义了两个异步函数:
没有调用,所以没有输出。
然后我们终于碰到了第一个输出:
输出结果第一条。
接下去调用async1函数,当调用async函数的时候会返回一个Promise对象。Promise对象是立即执行的。
那就执行函数第一行的输出:
输出结果第二条。
然后就遇到了await async2,注意,这里的机制就是在async里遇到await,它会使async1函数暂停执行,这一步会执行到await的后面的内容,也就是执行async2后就暂停执行了。,async1就被扔到异步队列了。
那执行async2呗:
输出结果第三条。
再往下碰到了Promise:
Promise对象是立即执行的,所以立即输出结果的第四条。而后是执行了resolve。执行成功,执行成功的话会走入promise的.then方法里,可是它是异步的回调函数,所以会被丢入到异步队列里。那么现在异步队列里是未执行完的async1和promise的.then方法。
接下去就碰到最后一个输出。
输出结果第五条。
这时候同步队列已经全部执行完成。开始执行异步队列里的任务。而队列是先进先出,所以先执行未执行完的async1。
输出结果第六条。
async1执行完毕,然后执行promise的.then方法,输出结果第七条。
最后再执行消息队列里的setTimeout,输出结果第八条。
通过断点调试可以很清晰的观察到结果的输出。
注:如果是两个计时器,就按上面说的被交给浏览器的定时器模块处理,当某个定时器到了可执行状态就会把该定时器扔到消息队列里去,最后执行。
结果:
下面的计时器虽然被后定义,但先到时,所以先被加入消息队列,自然就被先执行。