先讲讲为什么要写这篇文章,我发现很多人对async/await的理解在不同的层次,大部分人对异步模型还在一个相对模糊的概念,很多人对异步模型甚至是有许多错误的看法,我觉得有必要写一篇文章来好好讲讲这个异步模型的await在你认为阻塞的时候干了什么?
很多人看到下面的代码,很多人理解为await不就是为了阻塞doSomething1和doSomething2的执行,让doSomething1完成后再执行doSomething2吗?看效果好像是这样,但事实真的就是这样吗?
async function run(mark) {
await doSomething1();
await doSomething2();
}
run('first') // 调用
实际上远远不仅仅是你所看到代码的那样,在第一个await之后实际上异步模型还可以去干非当前所看到代码的事情。
这里你可能还不理解这句话,为什么叫“干非当前所看到代码的事情”?
下面这个例子可能可以更让你理解为什么叫“干非当前所看到代码的事情”,那我们略作修改,比如
function doSomething1 (time) {
return new Promise ((reslove)=>{setTimeout(reslove,time)})
}
async function run(mark) {
await doSomething1(1000);
console.log(`${mark}.doSomething1 complete`)
await doSomething2();
}
run('first') // 第一次调用
run('second') // 第二次调用
这时你会发现如果doSomething1的耗时比较固定的话,你会发现两条'doSomething1 complete'几乎会同时输出。为什么呢?如果按照await是阻塞的说法来说,不是应该走完第一次调用的doSomething1和doSomething2才会走下一步么?显然await是阻塞的说话是错误的。那这个不是阻塞是什么呢?我说这是一个调度系统,你相信么?
这就好比你是一个工人:
- 当第一次run()执行的时候,你运行到doSomething1发现有await,你就知道这里有需要等待的事情,好比你砌墙发现没砖了,你要等货车把砖搬过来。
- 所以这时候你跟调度系统说“现在第一次run()的doSomething1要等某样东西才能运行了”,这时候调度系统就会告诉你“那你现在去做第二次run()的doSomething1吧”
- 然后你去就做了第二次run()的doSomething1。同样发现第二次run()的doSomething1也需要等待.
- 你跟调度系统说“现在第二次run()的doSomething1要等某样东西才能运行了”。调度系统发现所有的任务都需要等待,这时可能就会跟你说“你先歇着,等我号令”。
- 过了一会调度系统告诉你第一次run()的doSomething1好了,快去看看,你做完了第一次run()的doSomething1后,就打印了'first.doSomething1 complete'。
- 接着运行第一次run()的doSomething2的时候发现要await。
- 你又跟调度系统说“现在第一次run()的doSomething2要等某样东西才能运行了”。
- 这时调度系统就会说“呀,第二次run()的doSomething1已经等你很久了,快去看看”。
- 这次你做完了第二次run()的doSomething1后,就打印了'second.doSomething1 complete'。由于中间要做的事情很少,所以两条'doSomething1 complete'几乎会同时输出。
这时你注意第2步,所以表面上你在第一次run()的doSomething1停住,但实际上他却跑去做第二次run()的doSomething1了,所以你认为他等待的时候,他并没有闲着。所以调度器让他去执行哪个方法就执行哪个方法,不一定就是当前await的方法。为什么要这样做?当然是为了更加高效去完成你要他完成的任务而不是一直在某个代码上发送空指令直到等到它完成。
同时这个调度系统还有个响亮的名字叫“事件循环”,事件循环在浏览器中分了宏微任务,在node.js还不仅仅宏微任务,还为事件循环分了六个阶段,同时一个微小的改动就会让事件循环的结果发生很大变化。比如本人之前写的文章《又被node的eventloop坑了,这次是node的锅》 可能大家在一些论坛上也看到过,这里说的是node11因为修改了对宏微任务的策略为了和浏览器趋同的故事。如果大家对异步调度器有兴趣可以去看更多关于事件循环的文章,因为本文主要讲解await是否是阻塞为主题,所以这里不作为重点讨论。
真正阻塞会让你一直陷在一件事情里面不能自拔,好比上面的代码,我们再做一个修改,如下:
function doSomething1 (time) {
return new Promise ((reslove)=>{setTimeout(reslove,time)})
}
async function run(mark) {
await doSomething1(1000);
console.log(`${mark}.doSomething1 complete`)
while(true){} // 增加一个死循环
await doSomething2();
}
run('first') // 第一次调用
run('second') // 第二次调用
这时你会发现,只会输出一条'first.doSomething1 complete',为什么呢?假设你还是那个工人还是上面那个故事,就是上面第5步之后让你有了做不完的事情,你就会在打印出'first.doSomething1 complete'后就会一直忙,所以没空去去问调度系统了。所以while(true)这会真正阻塞js进程,导致事件循环无法继续下去或者让事件循环产生极大延迟,这才是真正阻塞而非await。
从上面的例子里我们知道await和真正阻塞是有本质上的区别。
在经历上面的故事后,我们知道在await后并不是傻等,浏览器的其他js代码更不会因为await之后就不执行了。恰恰相反,await非常聪明,当它知道某件事情需要等待的时候他会去做一些别的不需要等待的事情,使得执行效率变得更高,极大减少CPU空转。所以请不要担心使用await会阻塞代码的问题了,今天你Get新知识了吗?谢谢大家。