javascript中事件循环(Event Loop)的执行顺序

如果下载的源码需要作者授权,请更换源码。本站免费分享资源不会增加授权

本篇文章给大家带来的内容是关于javascript中事件循环(Event Loop)的执行顺序,有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。

简要介绍:谈谈promise.resove,setTimeout,setImmediate,process.nextTick在EvenLoop队列中的执行顺序

问题的来源

event loop都不陌生,是指主线程从“任务队列”中循环读取任务,比如

例1:  setTimeout(function(){console.log(1)},0);  console.log(2)  //输出2,1

在上述的例子中,我们明白首先执行主线程中的同步任务,当主线程任务执行完毕后,再从event loop中读取任务,因此先输出2,再输出1。

event loop读取任务的先后顺序,取决于任务队列(Job queue)中对于不同任务读取规则的限定。比如下面一个例子:

例2:  setTimeout(function () {   console.log(3); }, 0);  Promise.resolve().then(function () {   console.log(2); }); console.log(1); //输出为  1  2 3

先输出1,没有问题,因为是同步任务在主线程中优先执行,这里的问题是setTimeout和Promise.then任务的执行优先级是如何定义的。

执行顺序

在Job queue中的队列分为两种类型:macro-task和microTask。我们举例来看执行顺序的规定,我们设

macro-task队列包含任务: a1, a2 , a3 micro-task队列包含任务: b1, b2 , b3

执行顺序为,首先执行marco-task队列开头的任务,也就是 a1 (a1代表同步的主任务)任务,执行完毕后,在执行micro-task队列里的所有任务,也就是依次执行b1, b2 , b3(异步),执行完后清空micro-task中的任务,接着执行marco-task中的第二个任务(异步),依次循环。

了解完了macro-task和micro-task两种队列的执行顺序之后,我们接着来看,真实场景下这两种类型的队列里真正包含的任务(我们以node V8引擎为例),在node V8中,这两种类型的真实任务顺序如下所示:

macro-task队列真实包含任务:

script(主程序代码)[对应上方的a1],setTimeout, setInterval, setImmediate, I/O, UI rendering

micro-task队列真实包含任务:

process.nextTick, Promises, Object.observe, MutationObserver

由此我们得到的执行顺序应该为:

script(主程序代码)—>process.nextTick—>Promises...——>setTimeout——>setInterval——>setImmediate——> I/O——>UI rendering

在ES6中macro-task队列又称为ScriptJobs,而micro-task又称PromiseJobs

举例

(1) setTimeout和promise

例3:  setTimeout(function () {   console.log(3); }, 0);  Promise.resolve().then(function () {   console.log(2); });  console.log(1);

(2) process.nextTick和promise、setTimeout

例子4: setTimeout(function(){console.log(1)},0);  new Promise(function(resolve,reject){    console.log(2);    resolve(); }).then(function(){console.log(3) }).then(function(){console.log(4)});  process.nextTick(function(){console.log(5)});  console.log(6); //输出2,6,5,3,4,1

(3)更复杂的例子

setTimeout(function(){console.log(1)},0);  new Promise(function(resolve,reject){    console.log(2);    setTimeout(function(){resolve()},0) }).then(function(){console.log(3) }).then(function(){console.log(4)});  process.nextTick(function(){console.log(5)});  console.log(6);  //输出的是  2 6 5 1 3 4

这些例子的原因请根据执行顺序执行判断,这里不一一解释了

本文由(壳先生)整理自网络,如转载请注明出处:https://www.mrshell.com;
本站发布的内容若侵犯到您的权益,请邮件联系 i@mrshell.com 删除,我们将及时处理!
===========================================================================

1. 本站大部分下载资源收集于网络,不保证其完整性以及安全性,请下载后自行测试。
2. 本站资源仅供学习和交流使用,版权归资源原作者所有,请在下载后24小时之内自觉删除。
3. 不得使用于非法商业用途,商用请支持正版!不得违反国家法律,否则后果自负!
4. 若作商业用途,请购买正版,由于未及时购买和付费发生的侵权行为,与本站无关。
5. 如有链接无法下载、失效或广告,请联系管理员处理!
6. 本站资源售价只是赞助,收取费用仅维持本站的日常运营所需!

=================================================================

壳先生 » javascript中事件循环(Event Loop)的执行顺序

发表评论

提供最优质的资源集合

立即查看 了解详情