深入分析JavaScript 事件循环(Event Loop)


Posted in Javascript onJune 19, 2020

事件循环(Event Loop),是每个JS开发者都会接触到的概念,但是刚接触时可能会存在各种疑惑。

众所周知,JS是单线程的,即同一时间只能运行一个任务。一般情况下这不会引发问题,但是如果我们有一个耗时较多的任务,我们必须等该任务执行完毕才能进入下一个任务,然而等待的这段时间常常让我们无法忍受,因为我们这段时间什么都不能做,包括页面也是锁死状态。

好在,时代在进步,浏览器向我们提供了JS引擎不具备的特性:Web API。Web API包括DOM API、定时器、HTTP请求等特性,可以帮助我们实现异步、非阻塞的行为。我们可以通过异步执行任务的方法来解决单线程的弊端,事件循环为此而生。

提问QAQ:为什么JavaScript是单线程的?

多个线程表示您可以同时独立执行程序的多个部分。确定一种语言是单线程还是多线程的最简单方法是看它拥有有多少个调用堆栈。JS 只有一个,所以它是单线程语言。

将JS设计为单线程是由其用途运行环境等因素决定的,作为浏览器脚本语言,JS的主要用途是与用户互动,以及操作DOM。这决定了它只能是单线程,否则会带来很复杂的同步问题。同时,单线程执行效率高。

1. Event Loop旧印象

大家熟悉的关于事件循环的机制说法大概是:主进程执行完了之后,每次从任务队列里取一个任务执行。如图所示,所有的任务分为同步任务和异步任务,同步任务直接进入任务队列-->主程序执行;异步任务则会挂起,等待其有返回值时进入任务队列从而被主程序执行。异步任务会通过任务队列的机制(先进先出的机制)来进行协调。具体如图所示:

深入分析JavaScript 事件循环(Event Loop)

同步和异步任务分别进入不同的执行环境,同步的进入主线程,即主执行栈,异步的进入任务队列。主线程内的任务执行完毕为空,会去任务队列读取对应的任务,推入主线程执行。 上述过程的不断重复就是我们所熟悉的Event Loop (事件循环)。但是promise出现之后,这个说法就不太准确了。

2. Event Loop 后印象

2.1 理论

这里首先用一张图展示JavaScript的事件循环:

深入分析JavaScript 事件循环(Event Loop)

直接看这张图,可能黑人问号已经出现在同学的脑海。。。

这里将task分为两大类,分别是macroTask(宏任务)和microTask(微任务).一次事件循环:先运行macroTask队列中的一个,然后运行microTask队列中的所有任务。接着开始下一次循环(只是针对macroTask和microTask,一次完整的事件循环会比这个复杂的多)。

那什么是macroTask?什么是microTask呢?

JavaScript引擎把我们的所有任务分门别类,一部分归为macroTask,另外一部分归为microTack,下面是类别划分:

macroTask:

  • setTimeout
  • setInterval
  • setImmediate
  • requestAnimationFrame
  • I/O
  • UI rendering

microTask:

  • process.nextTick
  • Promise
  • Object.observe
  • MutationObserver

我们所熟悉的定时器就属于macroTask,仅仅了解macroTask的机制还是不够的。为直观感受两种队列的区别,下面上代码进行实践感知。

2.2 实践

以setTimeout、process.nextTick、promise为例直观感受下两种任务队列的运行方式。

console.log('main1');

process.nextTick(function() {
 console.log('process.nextTick1');
});

setTimeout(function() {
 console.log('setTimeout');
 process.nextTick(function() {
 console.log('process.nextTick2');
 });
}, 0);

new Promise(function(resolve, reject) {
 console.log('promise');
 resolve();
}).then(function() {
 console.log('promise then');
});

console.log('main2');

别着急看答案,先以上面的理论自己想想,运行结果会是啥?

最终结果是这样的:

main1
promise
main2
process.nextTick1
promise then

// 第二次事件循环
setTimeout
process.nextTick2

process.nextTick 和 promise then在 setTimeout 前面输出,已经证明了macroTask和microTask的执行顺序。但是有一点必须要指出的是。上面的图容易给人一个错觉,就是主进程的代码执行之后,会先调用macroTask,再调用microTask,这样在第一个循环里一定是macroTask在前,microTask在后。

但是最终的实践证明:在第一个循环里,process.nextTick1和promise then这两个microTask是在setTimeout这个macroTask里之前输出的,这是因为Promises/A+规范规定主进程的代码也属于macroTask。

主进程这个macroTask(也就是main1、promise和main2)执行完了,自然会去执行process.nextTick1和promise then这两个microTask。这是第一个循环。之后的setTimeout和process.nextTick2属于第二个循环

别看上面那段代码好像特别绕,把原理弄清楚了,都一样 ~

requestAnimationFrame、Object.observe(已废弃) 和 MutationObserver这三个任务的运行机制大家可以从上面看到,不同的只是具体用法不同。重点说下UI rendering。在HTML规范:event-loop-processing-model里叙述了一次事件循环的处理过程,在处理了macroTask和microTask之后,会进行一次Update the rendering,其中细节比较多,总的来说会进行一次UI的重新渲染。

3. 小结

总而言之,记住一次事件循环:先运行macroTask队列中的一个,然后运行microTask队列中的所有任务。接着开始下一次循环。

参考文献:

  • JavaScript Event Loop相关原理解析
  • 深入理解事件循环机制
  • JavaScript运行机制

以上就是深入分析JavaScript 事件循环(Event Loop)的详细内容,更多关于JavaScript 事件循环(Event Loop)的资料请关注三水点靠木其它相关文章!

Javascript 相关文章推荐
通过ifame指向的页面高度调整iframe的高度
Oct 05 Javascript
javascript实现的动态文字变换
Jul 28 Javascript
子窗口、父窗口和Silverlight之间的相互调用
Aug 16 Javascript
js和html5实现手机端刮刮卡抽奖效果完美兼容android/IOS
Nov 18 Javascript
TypeScript学习之强制类型的转换
Dec 27 Javascript
Bootstrap媒体对象学习使用
Mar 07 Javascript
jQuery+Ajax请求本地数据加载商品列表页并跳转详情页的实现方法
Jul 12 jQuery
js 显示日期时间的实例(时间过一秒加1)
Oct 25 Javascript
vue生命周期的探索
Apr 03 Javascript
微信小程序的授权实现过程解析
Aug 02 Javascript
Vue SPA 初次进入加载动画实现代码
Nov 14 Javascript
angular *Ngif else用法详解
Dec 15 Javascript
原生JS实现微信通讯录
Jun 18 #Javascript
vue+element-ui表格封装tag标签使用插槽
Jun 18 #Javascript
js实现ajax的用户简单登入功能
Jun 18 #Javascript
JS实现躲避粒子小游戏
Jun 18 #Javascript
html-webpack-plugin修改页面的title的方法
Jun 18 #Javascript
vue实现购物车结算功能
Jun 18 #Javascript
vue-cli4.x创建企业级项目的方法步骤
Jun 18 #Javascript
You might like
php htmlspecialchars()与shtmlspecialchars()函数的深入分析
2013/06/05 PHP
php实现的zip文件内容比较类
2014/09/24 PHP
PHP pthreads v3下worker和pool的使用方法示例
2020/02/21 PHP
fckeditor 获取文本框值的实现代码
2009/02/09 Javascript
JS+CSS实现的竖向简洁折叠菜单效果代码
2015/10/22 Javascript
node.js入门实例helloworld详解
2015/12/23 Javascript
JavaScript中的函数(二)
2015/12/23 Javascript
使用javascript插入样式
2016/03/14 Javascript
对javascript继承的理解
2016/10/11 Javascript
Javascript将字符串日期格式化为yyyy-mm-dd的方法
2016/10/27 Javascript
详解Angular.js的$q.defer()服务异步处理
2016/11/06 Javascript
JavaScript基于自定义函数判断变量类型的实现方法
2016/11/23 Javascript
Angular实现响应式表单
2017/08/04 Javascript
详解vue渲染函数render的使用
2017/12/12 Javascript
小程序实现选择题选择效果
2018/11/04 Javascript
vue实现鼠标移入移出事件代码实例
2019/03/27 Javascript
微信小程序 自定义复选框实现代码实例
2019/09/04 Javascript
vue实现移动端项目多行文本溢出省略
2020/07/29 Javascript
vue实现按钮切换图片
2021/01/20 Vue.js
使用python BeautifulSoup库抓取58手机维修信息
2013/11/21 Python
Python实现的堆排序算法示例
2018/04/29 Python
transform python环境快速配置方法
2018/09/27 Python
基于Python数据分析之pandas统计分析
2020/03/03 Python
Python爬虫如何破解JS加密的Cookie
2020/11/19 Python
澳大利亚相机之家:Camera House
2017/11/30 全球购物
Johnston & Murphy官网: 约翰斯顿·墨菲牛津总统鞋
2018/01/09 全球购物
Java中compareTo和compare的区别
2016/04/12 面试题
自我评价中英文语句
2013/11/30 职场文书
《郑和远航》教学反思
2014/04/16 职场文书
校庆标语集锦
2014/06/25 职场文书
庆六一活动总结
2014/08/29 职场文书
吧主申请感言怎么写
2015/08/03 职场文书
奖学金主要事迹范文
2015/11/04 职场文书
SpringCloud Feign请求头删除修改的操作代码
2022/03/20 Java/Android
Python实现批量将文件复制到新的目录中再修改名称
2022/04/12 Python
Mysql 如何合理地统计一个数据库里的所有表的数据量
2022/04/18 MySQL