详解JS浏览器事件循环机制


Posted in Javascript onMarch 27, 2019

先来明白些概念性内容。

进程、线程

进程是系统分配的独立资源,是 CPU 资源分配的基本单位,进程是由一个或者多个线程组成的。

线程是进程的执行流,是CPU调度和分派的基本单位,同个进程之中的多个线程之间是共享该进程的资源的。

浏览器内核

浏览器是多进程的,浏览器每一个 tab 标签都代表一个独立的进程(也不一定,因为多个空白 tab 标签会合并成一个进程),浏览器内核(浏览器渲染进程)属于浏览器多进程中的一种。

浏览器内核有多种线程在工作。

GUI 渲染线程:

  1. 负责渲染页面,解析 HTML,CSS 构成 DOM 树等,当页面重绘或者由于某种操作引起回流都会调起该线程。
  2. 和 JS 引擎线程是互斥的,当 JS 引擎线程在工作的时候,GUI 渲染线程会被挂起,GUI 更新被放入在 JS 任务队列中,等待 JS 引擎线程空闲的时候继续执行。

JS 引擎线程:

  1. 单线程工作,负责解析运行 JavaScript 脚本。
  2. 和 GUI 渲染线程互斥,JS 运行耗时过长就会导致页面阻塞。

事件触发线程:

当事件符合触发条件被触发时,该线程会把对应的事件回调函数添加到任务队列的队尾,等待 JS 引擎处理。

定时器触发线程:

  1. 浏览器定时计数器并不是由 JS 引擎计数的,阻塞会导致计时不准确。
  2. 开启定时器触发线程来计时并触发计时,计时完成后会被添加到任务队列中,等待 JS 引擎处理。

http 请求线程:

  1. http 请求的时候会开启一条请求线程。
  2. 请求完成有结果了之后,将请求的回调函数添加到任务队列中,等待 JS 引擎处理。

 详解JS浏览器事件循环机制

JavaScript 引擎是单线程

JavaScript 引擎是单线程,也就是说每次只能执行一项任务,其他任务都得按照顺序排队等待被执行,只有当前的任务执行完成之后才会往下执行下一个任务。

HTML5 中提出了 Web-Worker API,主要是为了解决页面阻塞问题,但是并没有改变 JavaScript 是单线程的本质。了解 Web-Worker。

JavaScript 事件循环机制

JavaScript 事件循环机制分为浏览器和 Node 事件循环机制,两者的实现技术不一样,浏览器 Event Loop 是 HTML 中定义的规范,Node Event Loop 是由 libuv 库实现。这里主要讲的是浏览器部分。

Javascript 有一个 main thread 主线程和 call-stack 调用栈(执行栈),所有的任务都会被放到调用栈等待主线程执行。

JS 调用栈

JS 调用栈是一种后进先出的数据结构。当函数被调用时,会被添加到栈中的顶部,执行完成之后就从栈顶部移出该函数,直到栈内被清空。

同步任务、异步任务

JavaScript 单线程中的任务分为同步任务和异步任务。同步任务会在调用栈中按照顺序排队等待主线程执行,异步任务则会在异步有了结果后将注册的回调函数添加到任务队列(消息队列)中等待主线程空闲的时候,也就是栈内被清空的时候,被读取到栈中等待主线程执行。任务队列是先进先出的数据结构。

Event Loop

调用栈中的同步任务都执行完毕,栈内被清空了,就代表主线程空闲了,这个时候就会去任务队列中按照顺序读取一个任务放入到栈中执行。每次栈内被清空,都会去读取任务队列有没有任务,有就读取执行,一直循环读取-执行的操作,就形成了事件循环。

详解JS浏览器事件循环机制

详解JS浏览器事件循环机制

定时器

定时器会开启一条定时器触发线程来触发计时,定时器会在等待了指定的时间后将事件放入到任务队列中等待读取到主线程执行。

定时器指定的延时毫秒数其实并不准确,因为定时器只是在到了指定的时间时将事件放入到任务队列中,必须要等到同步的任务和现有的任务队列中的事件全部执行完成之后,才会去读取定时器的事件到主线程执行,中间可能会存在耗时比较久的任务,那么就不可能保证在指定的时间执行。

宏任务(macro-task)、微任务(micro-task)

除了广义的同步任务和异步任务,JavaScript 单线程中的任务可以细分为宏任务和微任务。

macro-task包括:script(整体代码), setTimeout, setInterval, setImmediate, I/O, UI rendering。

micro-task包括:process.nextTick, Promises, Object.observe, MutationObserver。

console.log(1);
setTimeout(function() {
  console.log(2);
})
var promise = new Promise(function(resolve, reject) {
  console.log(3);
  resolve();
})
promise.then(function() {
  console.log(4);
})
console.log(5);

示例中,setTimeout 和 Promise被称为任务源,来自不同的任务源注册的回调函数会被放入到不同的任务队列中。

有了宏任务和微任务的概念后,那 JS 的执行顺序是怎样的?是宏任务先还是微任务先?

第一次事件循环中,JavaScript 引擎会把整个 script 代码当成一个宏任务执行,执行完成之后,再检测本次循环中是否寻在微任务,存在的话就依次从微任务的任务队列中读取执行完所有的微任务,再读取宏任务的任务队列中的任务执行,再执行所有的微任务,如此循环。JS 的执行顺序就是每次事件循环中的宏任务-微任务。

  1. 上面的示例中,第一次事件循环,整段代码作为宏任务进入主线程执行。
  2. 遇到了 setTimeout ,就会等到过了指定的时间后将回调函数放入到宏任务的任务队列中。
  3. 遇到 Promise,将 then 函数放入到微任务的任务队列中。
  4. 整个事件循环完成之后,会去检测微任务的任务队列中是否存在任务,存在就执行。
  5. 第一次的循环结果打印为: 1,3,5,4。
  6. 接着再到宏任务的任务队列中按顺序取出一个宏任务到栈中让主线程执行,那么在这次循环中的宏任务就是 setTimeout 注册的回调函数,执行完这个回调函数,发现在这次循环中并不存在微任务,就准备进行下一次事件循环。
  7. 检测到宏任务队列中已经没有了要执行的任务,那么就结束事件循环。
  8. 最终的结果就是 1,3,5,4,2。

以上所述是小编给大家介绍的JS浏览器事件循环机制详解整合,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对三水点靠木网站的支持!

Javascript 相关文章推荐
WordPress 插件——CoolCode使用方法与下载
Jul 02 Javascript
零基础学JavaScript最新动画教程+iso光盘下载
Jan 22 Javascript
javascript中的继承实例代码
Apr 27 Javascript
使用jQuery设置disabled属性与移除disabled属性
Aug 21 Javascript
jQuery中:checkbox选择器用法实例
Jan 03 Javascript
实现placeholder效果的方案汇总
Jun 11 Javascript
基于JS实现的笛卡尔乘积之商品发布
May 13 Javascript
浅谈JavaScript中小数和大整数的精度丢失
May 31 Javascript
JavaScript中数组slice和splice的对比小结
Sep 22 Javascript
利用Bootstrap Multiselect实现下拉框多选功能
Apr 08 Javascript
JS匿名函数内部this指向问题详析
May 10 Javascript
使用 Opentype.js 生成字体子集的实例代码详解
May 25 Javascript
详解如何更好的使用module vuex
Mar 27 #Javascript
原生js实现获取form表单数据代码实例
Mar 27 #Javascript
JQueryDOM之样式操作
Mar 27 #jQuery
Element中的Cascader(级联列表)动态加载省\市\区数据的方法
Mar 27 #Javascript
Node.js 多线程完全指南总结
Mar 27 #Javascript
浅谈JS和jQuery的区别
Mar 27 #jQuery
Nginx设置为Node.js的前端服务器方法总结
Mar 27 #Javascript
You might like
手把手教你使用DedeCms V3的在线采集图文教程
2007/04/03 PHP
PHP实现基于状态的责任链审批模式详解
2019/05/31 PHP
如何在Laravel5.8中正确地应用Repository设计模式
2019/11/26 PHP
javascript 面向对象编程 function也是类
2009/09/17 Javascript
Javascript异步编程的4种方法让你写出更出色的程序
2013/01/17 Javascript
Jquery Ajax方法传值到action的方法
2014/05/11 Javascript
jQuery实现批量判断表单中文本框非空的方法(2种方法)
2015/12/09 Javascript
canvas实现图片根据滑块放大缩小效果
2017/02/24 Javascript
Bootstrap Table使用整理(五)之分页组合查询
2017/06/09 Javascript
详解Node.js access_token的获取、存储及更新
2017/06/20 Javascript
解决vue-quill-editor上传内容由于图片是base64的导致字符太长的问题
2018/08/20 Javascript
关于JavaScript中高阶函数的魅力详解
2018/09/07 Javascript
JS中验证整数和小数的正则表达式
2018/10/08 Javascript
微信小程序全局变量功能与用法详解
2019/01/22 Javascript
vue-cli 3 全局过滤器的实例代码详解
2019/06/03 Javascript
浅谈JS中this在各个场景下的指向
2019/08/14 Javascript
Javascript Web Worker使用过程解析
2020/03/16 Javascript
Vue-cli打包后如何本地查看的操作
2020/09/02 Javascript
[02:01]BBC DOTA2国际邀请赛每日综述:八强胜者组鏖战,中国队喜忧参半
2014/07/19 DOTA
动态创建类实例代码
2009/10/07 Python
python获得文件创建时间和修改时间的方法
2015/06/30 Python
python实现画一颗树和一片森林
2018/06/25 Python
详解python里的命名规范
2018/07/16 Python
django小技巧之html模板中调用对象属性或对象的方法
2018/11/30 Python
python redis连接 有序集合去重的代码
2019/08/04 Python
python itsdangerous模块的具体使用方法
2020/02/17 Python
PyCharm中配置PySide2的图文教程
2020/06/18 Python
python re.match()用法相关示例
2021/01/27 Python
Lacoste(法国鳄鱼)加拿大官网:以标志性的POLO衫而闻名
2019/05/15 全球购物
如何实现一个自定义类的序列化
2012/05/22 面试题
五年级科学教学反思
2014/02/05 职场文书
党员查摆问题及整改措施
2014/10/10 职场文书
劳模事迹材料范文
2014/12/24 职场文书
英文邀请函
2015/02/02 职场文书
《火纹风花雪月无双》预告“神秘雇佣兵” 紫发剑客
2022/04/13 其他游戏
SQL Server #{}可以防止SQL注入
2022/05/11 SQL Server