谈一谈浏览器与Node.js中的JavaScript事件循环,宏任务与微任务机制

2023-10-28

JavaScript中的异步代码

JavaScript是一个单线程非阻塞的脚本语言。这代表代码是执行在一个主线程上面的。但是JavaScript中有很多耗时的异步操作,例如AJAX,setTimeout等等;也有很多事件,例如用户触发的点击事件,鼠标事件等等。这些异步操作并不会阻塞我们代码的执行。例如:

let a = 1;
setTimeout(() => {
  console.log('->', a)
}, 10);
a = 2;
// 输出 -> 2

可以看到,上述代码在浏览器中执行时,遇到setTimeout操作,并没有阻塞等待异步操作的结束再继续执行代码,而是先继续执行后面的代码。等异步操作结束后,浏览器再回来执行异步回调中的代码。因此,上述代码的console.log输出时,a的值已经变为了2。

这些异步非阻塞的实现,就是靠Javascript中的事件循环机制。

JavaScript中的线程

上面说到JavaScript是一个单线程的语言,这句话并不完全对。单线程指的是代码在一个主线程中运行,但是代码所触发的任务不一定在主线程运行。除了执行代码的线程之外,执行JavaScript的环境中还包含其他很多线程。其中浏览器的线程与Node.js中的线程也不相同。

浏览器中的线程

注意,这里对于浏览器线程进行了抽象和总结。实际上浏览器的线程和进程要更复杂,而且有时候会根据浏览器版本的不同而变化,因此仅供参考。

  • JS主线程
    负责运行JavaScript代码,解析HTML,CSS,构建DOM树,布局和绘制页面等等。
  • 事件监听线程
    负责监听触发的各种事件,放入事件循环中。
  • HTTP请求线程
    负责处理各类网络请求。
  • 定时触发器线程
    为setInterval,setTimeout定时触发操作等操作进行定时计数的线程。

浏览器中的进程

上面的线程实际上都在浏览器中的渲染进程中包含。一个浏览器要想正常运行,只做上述的操作是不够的。我们以Chrome为例,列举一个浏览器运行所需要的进程。

  • 浏览器进程
    负责网页外的界面功能,例如地址栏,书签等等。
  • GPU进程
    负责使用GPU渲染界面。
  • 网络进程
    负责网络相关的请求处理。
  • 插件进程
    负责浏览器插件运行。
  • 渲染进程
    负责网页内页面展示相关的操作,即上一节浏览器中的线程包含的所有线程都在这个进程中执行。

一个浏览器可以拥有多个标签页,在不同的标签页中,除了渲染进行之外,都是共享的。即我们打开一个新的标签页时,会产生一个新的渲染进程。(当在原标签页中打开新标签页,且属于同一个域则共享一个渲染进程)

进程与线程的关系

上面我们了解了浏览器中的进程和线程,有些同学就会有疑问,为什么要设立这么多的进程和线程?

进程是操作系统分配资源的基本单位,而线程是CPU任务调度和执行的基本单位。

简单理解下就是一个完整的应用程序是以进程为单位的,即至少有一个进程。而一段程序/代码在CPU的独立执行则至少以线程为单位。不同的进程和不同的线程都可以并行运行。

一个进程可以包含很多个线程,多个线程共享一个进程的资源(比如内存)。当一个进程崩溃后不会影响其他进程,但是当一个线程崩溃,它所在的整个进程都会崩溃掉,这个进程内的其他线程也会崩溃。

因此,为了同时并行执行代码和异步请求,浏览器中的渲染进程包含很多线程来并行运行任务。而为了让不同标签页的网页不互相影响,不同标签页拥有独立的渲染进程。这样即使某个网页崩溃,也不会影响其他标签页。

Node.js中的线程

  • JS主线程
    负责运行JavaScript代码。
  • libuv的异步I/O线程池
    负责实现事件循环和异步IO等操作,在不同操作系统的具体实现方式不同。
  • 用户创建的线程

上述这些进程和线程的说明也仅仅是进行了抽象和简化,事实上浏览器和Node.js中的进程和线程数要更多,处理也更复杂。

宏任务与微任务

Javascript中的异步任务大致可以分为两种:宏任务和微任务。宏任务和微任务的执行顺序和优先级是不同的,具体的执行顺序问题我们在事件循环中描述,这里先来看一下,哪些操作属于宏任务,哪些属于微任务。这里仅仅是简单介绍,更详细的要在了解事件循环之后说明。

宏任务

任务 浏览器 Node.js 描述
setTimeout 在指定的毫秒数后调用函数
setInterval 定时调用函数
script标签 整体代码块
I/O请求 例如文件请求,网络请求等
DOM事件 例如点击事件,hover事件等
requestAnimationFrame 浏览器重绘前更新动画
postMessage iframe跨域通信
MessageChannel 管道通信
setImmediate 一次事件循环执行完毕调用

微任务

任务 浏览器 Node.js 描述
Promise中resolve和reject回调
async函数中的await异步函数
MutationObserver 监听DOM变动触发
process.nextTick 当前任务结束后执行

事件循环

与上面进程与线程的介绍一样,在浏览器中与Node.js中实现循环的方式也并不相同。下面我们来分别简单介绍一下。注意,这仅仅是对执行逻辑的抽象和总结,实际上浏览器和Node.js中的实现要更复杂。

浏览器中的事件循环

浏览器中的事件循环可以分为两个队列,宏任务队列和微任务队列。具体的任务执行顺序如下:

  1. 解析HTML中遇到script标签,开始执行第一个宏任务。
  2. 在宏任务执行中遇到宏任务,执行其中的请求(例如网络请求,定时器),在请求完成后将回调放入宏任务队列中。
  3. 在宏任务执行中遇到微任务,暂不执行回调,而是放入微任务队列中。
  4. 宏任务执行完成。开始依次执行微任务队列中的任务。
  5. 微任务执行中遇到宏任务或者微任务,处理方式同上,分别放入各自的队列中。
  6. 微任务队列清空后,开始执行宏任务队列中的下一个任务。

在事件循环的流程中,微任务的优先级实际上更高,执行完一个宏任务之后,要执行微任务队列中的所有任务。

为什么要区分宏任务和宏任务,优先级也不同

因为不同任务的开销不同,有的任务需要调用不同的线程甚至进程,有的任务需要等待请求返回甚至定时。

  1. 如果将全部的任务同步执行,那些耗时较久的任务会阻塞,造成整个页面加载缓慢。假设有请求A耗时10秒,请求B耗时20秒,如果同步执行,需要耗费30秒。如果将请求由其它线程实现,回调放入宏任务,则执行流程变为:执行代码->碰到A请求,其他线程异步等待返回->继续执行代码->碰到b请求,其他线程异步等待返回。A和B就实现了异步请求,回调被分别放入宏任务,等待下次事件循环。耗时间为20秒。
  2. 为什么微任务的优先级更高?因为微任务大部分是耗时不太久,不需要等待其他线程/进程等待完成通知的。因此,微任务相当于在宏任务的基础上进行了“插队”,拥有更高的优先级,也提高了页面的响应速度。

为什么script标签是宏任务呢?

  1. script标签可能需要异步请求获取,例如<script src="myscripts.js"></script>
  2. script标签是嵌入在HTML中的,浏览器需要将HTML中的script标签解析出来供执行,这个步骤需要耗费一定的时间。

浏览器事件循环的更多说明

WHATWG(网页超文本应用技术工作小组)在官网对事件循环和任务队列做出了更详细的说明和解释,可以作为参考:说明文档。在新的说明中,任务的分类和事件循环已经有了部分区别,这里简要说一下,更多还请直接查看文档:

  1. 事件循环不一定对应于多线程。例如多个事件循环可以在单个线程中协作调度。
  2. 任务队列并不是一个严格的队列,而是一个集合。每次从队列中取出一个可以被执行的任务,而不是选取第一个任务(可能该任务还在阻塞中)。
  3. 宏任务队列有多个,不同类型的任务(任务源)放置在不同的任务队列中。具体的选取规则浏览器根据实际情况确定。

Node.js中的宏任务队列

Node.js的官网给出了事件循环的文档。它的事件循环要比浏览器的看起来复杂一些。下面是Node.js的宏任务队列。

   ┌───────────────────────────┐
┌─>│           timers          │
│  └─────────────┬─────────────┘
│  ┌─────────────┴─────────────┐
│  │     pending callbacks     │
│  └─────────────┬─────────────┘
│  ┌─────────────┴─────────────┐
│  │       idle, prepare       │
│  └─────────────┬─────────────┘      ┌───────────────┐
│  ┌─────────────┴─────────────┐      │   incoming:   │
│  │           poll            │<─────┤  connections, │
│  └─────────────┬─────────────┘      │   data, etc.  │
│  ┌─────────────┴─────────────┐      └───────────────┘
│  │           check           │
│  └─────────────┬─────────────┘
│  ┌─────────────┴─────────────┐
└──┤      close callbacks      │
   └───────────────────────────┘

Node.js的宏任务队列并不是一整个队列,而是根据事件类型做出了区分,分为了六个队列,依次执行:

  1. timers 定时器队列,执行定时器的回调
  2. pending callbacks 挂起的回调函数,用于某些系统回调
  3. idle, prepare 仅在内部使用
  4. poll 执行I/O事件回调
  5. check setImmediate回调
  6. close callbacks close事件的回调,例如 socket.on('close', ...)

其中我们的大部分宏任务回调都会在poll阶段执行,除了timerscheckclose callbacks阶段的特殊回调。每个宏任务队列都有自己的微任务队列。

在这里插入图片描述

Node.js事件循环的流程

  1. 首先执行主线代码,遇到宏任务就分配到对应的宏任务队列中,微任务也划分到主线的微任务队列中,直到执行完毕。
  2. 执行主线代码的微任务队列中的所有任务。
  3. 没有宏任务则执行结束,有则开始事件循环。在事件循环中,按照上述的6个宏任务队列依次执行。下面的步骤是单个队列中的流程。
  4. 在单个宏任务队列中,选择一个宏任务执行。如果执行中遇到新的宏任务就分配到对应的宏任务队列中。遇到微任务就放到该宏任务的微任务队列中。
  5. 一个宏任务执行完毕后,执行process.nextTick中的回调(如果有)。
  6. 执行当前宏任务的微任务队列中的任务,直到微任务队列清空。
  7. 在上面的单个宏任务队列中,再选择一个宏任务执行。直到当前宏任务队列清空或者到达上限。
  8. 选择下一个宏任务队列执行。

6个宏任务队列都执行完毕,才叫做一次事件循环执行完毕。

Node.js的11版本之前的区别

其中,在Node.js的11版本之前,宏任务和微任务的执行关系与上述流程不同:

每个宏任务队列有一个微任务队列。在单个宏任务队列中,首先执行完所有的宏任务,如果遇到微任务就放到微任务队列中。当单个宏任务队列中的所有宏任务执行完毕后,再执行该宏任务队列的微任务队列。

对比执行流程的区别,可以看到Node.js的11版本提高了微任务队列中的优先级,让Node.js中微任务队列的优先级和浏览器中的表现类似。而process.nextTick可以看做是一个比微任务更高优先级的钩子。

注意

  • setTimeout的时间即使设置为0,也会有一个最小时间,因此它与setImmediate谁更早执行不一定。
  • 并不是所有回调函数都是异步的。例如new Promise(fun)中的回调是同步执行,在回调中遇到resolve(), reject()等才是微任务异步执行的。

参考

  • JavaScript 之事件循环 (Event Loop)
    https://xie.infoq.cn/article/921841837025748baac847030
  • The Node.js Event Loop, Timers, and process.nextTick()
    https://nodejs.org/en/docs/guides/event-loop-timers-and-nexttick
    https://nodejs.org/zh-cn/docs/guides/event-loop-timers-and-nexttick
  • 深入理解浏览器中的进程与线程
    https://juejin.cn/post/6991849728493256741
  • 这一篇浏览器事件循环,可能会颠覆部分人的对宏任务和微任务的理解
    https://juejin.cn/post/7259927532249710653
  • HTML Living Standard (event-loops)
    https://html.spec.whatwg.org/multipage/webappapis.html#event-loops
  • 阿里一面:熟悉事件循环?那谈谈为什么会分为宏任务和微任务
    https://juejin.cn/post/7073099307510923295
  • node.js事件循环简单理解——定时器,process.nextTick()等
    https://blog.csdn.net/qq_46561394/article/details/123172336
  • 手摸手带你彻底掌握,任务队列、事件循环、宏任务、微任务
    https://juejin.cn/post/6979876135182008357
  • 浏览器UI线程和JS线程是同一个线程吗?
    https://www.zhihu.com/question/264253488
  • 微信小程序的双线程设计有何创新之处?浏览器的渲染线程和 JS 线程本来不就是两个独立线程吗?
    https://www.zhihu.com/question/446103629
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

谈一谈浏览器与Node.js中的JavaScript事件循环,宏任务与微任务机制 的相关文章

  • 解析“流”JSON

    我在浏览器中有一个网格 我想通过 JSON 将数据行发送到网格 但浏览器应该在接收到 JSON 时不断解析它 并在解析时将行添加到网格中 换句话说 在接收到整个 JSON 对象后 不应将行全部添加到网格中 应该在接收到行时将其添加到网格中
  • 使用 JavaScript 使链接保持活动状态并在单击时显示悬停效果

    I am struggling to make this work I d like to make it where if O F is clicked the hover state stays active if another li
  • 无法通过节点应用程序连接到redis,两者都在docker中

    我正在尝试将我的应用程序连接到 redis 但我得到 ioredis Unhandled error event Error connect ECONNREFUSED 127 0 0 1 6379 当我做 docker exec it ed
  • Jquery/Javascript 上传和下载文件,无需后端

    是否可以在没有后端服务器的情况下在 JavaScript 函数中下载和上传文件 我需要导出和导入由 JavaScript 函数生成的 XML 我想创建按钮 保存 xml 来保存文件 但我不知道是否可行 另一方面 我希望将 XML 文件直接上
  • Meteor - 从客户端取消服务器方法

    我正在通过服务器方法执行数据库计数 用户可以选择他们希望如何执行计数 然后调用该方法 我的问题是 计数可能需要一些时间 并且用户可能会在方法运行时改变主意并请求不同的计数 有什么方法可以取消调用的方法并运行新的计数吗 我认为 this un
  • 在javascript中解析json - 长数字被四舍五入

    我需要解析一个包含长数字的 json 在 java servlet 中生成 问题是长数字被四舍五入 当执行这段代码时 var s x 6855337641038665531 var obj JSON parse s alert obj x
  • 是否有“npmpublish-f”的解决方法

    现在npm publish f已弃用 是否有解决方法或软件包可以覆盖发布后的目标版本 我知道关于semver http semver org 我还想要npm publish f 您可以取消发布特定版本 然后重新发布它 npm unpubli
  • 通过 CDN 使用 Dojo 时如何加载自定义 AMD 模块?

    我正在使用 google 的 CDN 并尝试使用他们的加载程序加载我自己的 AMD 模块 我知道我做错了什么 但我被困住了 有任何想法吗
  • 如何在react-native中获取Text组件的onPress值

    我是一名新的 React Native 开发人员 我想使用 onPress 获取 Text 组件的值并将其传递给函数
  • 如何获取 vuejs 组件单元测试中定义的“this”变量

    我正在尝试在 npm 脚本中使用 mocha webpack 来测试 vuejs 组件 我在测试中像这样嘲笑 vuex 商店 const vm new Vue template div div
  • 提交表单并重定向页面

    我在 SO 上看到了很多与此相关的其他问题 但没有一个对我有用 我正在尝试提交POST表单 然后将用户重定向到另一个页面 但我无法同时实现这两种情况 我可以获取重定向或帖子 但不能同时获取两者 这是我现在所拥有的
  • 无法运行 npm install

    In here http devdocs magento com guides v2 0 frontend dev guide css topics css debug html它说要跑npm install 但是当我运行时出现此错误sud
  • 为什么在 Internet Explorer 中访问 localStorage 对象会引发错误?

    我正在解决一个客户端问题 Modernizr 意外地没有检测到对localStorageInternet Explorer 9 中的对象 我的页面正确使用 HTML 5 文档类型 并且开发人员工具报告该页面具有 IE9 的浏览器模式和 IE
  • 模块构建失败(来自 ./node_modules/babel-loader/lib/index.js)Vue Js

    我从 GitHub 下载了一个我和我的朋友正在开发的项目 但是当我尝试运行时 npm run serve 我收到这个错误 src main js 中的错误 Module build failed from node modules babe
  • Javascript转换时区问题

    我在转换当前时区的日期时间时遇到问题 我从服务器收到此日期字符串 格式为 2015 10 09T08 00 00 这是中部时间 但是当我使用 GMT 5 中的 new Date strDate 转换此日期时间时 它返回给我的信息如下 这是不
  • Safari 支持 JavaScript window.onerror 吗?

    我有一个附加到 window onerror 的函数 window onerror function errorMsg url line window alert asdf 这在 firefox chrome 和 IE 中工作正常 但在 s
  • JQuery 图像上传不适用于未来的活动

    我希望我的用户可以通过帖子上传图像 因此 每个回复表单都有一个上传表单 用户可以通过单击上传按钮上传图像 然后单击提交来提交帖子 现在我的上传表单可以上传第一个回复的图像 但第二个回复的上传不起作用 我的提交过程 Ajax 在 php 提交
  • 如何获取浏览器视口中当前显示的内容

    如何获取当前正在显示长文档的哪一部分的指示 例如 如果我的 html 包含 1 000 行 1 2 3 9991000 并且用户位于显示第 500 行的中间附近 那么我想得到 500 n501 n502 或类似的内容 显然 大多数场景都会比
  • 导致回发到与弹出窗口不同的页面

    我有一个主页和一个详细信息页面 详细信息页面是从主页调用的 JavaScript 弹出窗口 当单击详细信息页面上的 保存 按钮时 我希望主页 刷新 是否有一种方法可以调用主页的回发 同时还可以从详细信息页面维护保存回发 Edit 使用win
  • 如何从图像输入中获取 xy 坐标?

    我有一个输入设置为图像类型

随机推荐

  • 全球十大即时通信软件最新排名

    第十名 Signal 1亿用户 Signal是一款提供加密通信的即时通讯软件 用户可以进行点对点的私密聊天和通话 优势 客户端及服务器开源 默认私聊 群聊端对端加密 纯净无广告 没有复杂功能 局限 注册使用Signal必须与手机捆绑 无法在
  • SQL数据库编写及示例

    一 数据库编写 1 数据库常用约束 主键约束 primary key 外键约束 foreign key references 唯一值约束 unique 默认值约束 default 检查约束 check 非空约束 not null 标识列 i
  • Apolo学习

    安装 java1 8 mysql 5 6 5以上 下载quickStart的包 早apollo下执行两个sql 如果不执行这两个sql apollo是执行不起来的 会有两个表来记录apollo的执行情况 其中一个表叫apolloportal
  • Sass语法(三)之循环

    一 数据类型 1 数字 如 1 2 13 10px 2 字符串 有俩种类型 a 有引号字符串 quoted strings 如 Lucida Grande http sass lang com b 无引号字符串 unquoted strin
  • C语言--指针:最底层的解释(慢慢懂~)

    指针 又可以叫它为地址 他表示的就是变量的被存储的地址 举个例子 创建整型变量a 于是a被存储到了内存中 我们就可以通过创建指针来找到他被存储到了哪个位置 取地址操作符 我们可以打印出a的指针 在这里 p为指针变量 指向a 不同于指针 打印
  • C++STL之vector与list

    文章目录 关于vector的用法 关于List的用法 vector和list的区别 关于vector的用法 include
  • C++ 享元模式

    什么是享元模式 享元模式是一种结构型设计模式 实现了在较少内存开销的同时 又支持了大量的对象 主要在资源有限的情况下 对创建大量对象行为的一种约束行为 享元模式的适用特征 当程序中有大量的相同对象 这些对象消耗了大量的内存 这些对象的状态可
  • centos7系统引导自动重启_Linux老鸟给出的Linux系统故障问题汇总,值得永久收藏...

    一 处理linux系统故障的思路 作为一名优秀的linux运维工程师 一定要有一套清晰 明确的解决故障思路 当问题出现时 才能迅速定位 解决问题 在开始本文学习之前 我根据多年工作和处理问题和故障的经验 总结出了一套处理问题的一般思路 供大
  • 全卷积网络FCN详细讲解(超级详细哦)

    原文链接 https blog csdn net qq 41760767 article details 97521397 depth 1 utm source distribute pc relevant none task utm so
  • Job thrrew an unhandled exception 是什么报错

    Job threw an unhandled exception 是一个通用的错误信息 它表明在执行一个作业 Job 时发生了未处理的异常 这个错误信息并不提供具体的异常信息 而只是告诉你作业执行过程中出现了未处理的异常 并且该异常没有被适
  • HSQLDB 介绍

    HSQLDB HSQL使用 java 语言编写的免费数据库 相对其他数据库 体积很小 是一个非常轻量级的数据库 不需要安装 而且支持嵌套查询和Indentity主键 下面结合在程序中使用 对 HSQL 作个简要的介绍和使用 一 简介 HSQ
  • ABAP DOI详解(1)

    什么是 DOI DOI 是Desktop Office Integration的缩写 是 SAP 提供的解决与 Office 集成的技术方案 早期 SAP 用 OLE 技术解决 与 Office 集成 OLE 语法参照 VBA 在 ABAP
  • 挂载cifs报错mount error(13): Permission denied

    Linux挂载Windows共享时 报以下错误 mount error 13 Permission deniedRefer to the mount cifs 8 manual page e g man mount cifs 解决方法 用户
  • 网络基本概念

    目录 一 IP地址 1 概念 2 格式 3 特殊IP 二 端口号 1 概念 2 格式 3 注意事项 三 协议 1 概念 2 作用 四 协议分层 1 网络设备所在分层 五 封装与分用 六 客户端和服务器 1 客户端与服务器通信的过程 一 IP
  • AWS EKS集群动态创建卷并挂载

    文章目录 AWS EKS动态创建卷配置 需求 EBS CSI DRIVER 什么是EBS CSI DRIVER 为什么要安装EBS CSI DRIVER 将 Amazon EBS CSI 驱动程序作为 Amazon EKS 附加组件管理 先
  • java 用webservice 获取国内手机号码归属地省份、地区和手机卡类型信息

    提供一个web服务网址供大家学习 http webservice webxml com cn WebServices WeatherWebService asmx 或者 http www webxml com cn zh cn web se
  • 两台Windows文件服务器同步,windows多服务器同步文件

    windows多服务器同步文件 内容精选 换一换 当创建文件系统后 您需要使用云服务器来挂载该文件系统 以实现多个云服务器共享使用文件系统的目的 本章节以Windows 2012版本操作系统为例进行CIFS类型的文件系统的挂载 同一SFS容
  • 差分电流采样电路解读

    电流差分采样电路如上图所示 这也是经典的电压差分采样电路 电流采样是通过一个小电阻 约几毫欧到几十毫欧之间 将电流信号转换成电压信号来进行采样 原理比较简单 但在实际测试过程中存在一些问题 现在对这些问题进行记录并解释 问题如下 1 采样电
  • 博途组态阀岛_SMC EX260总线阀岛

    在博途中组态SMC EX260 SPN根据SMC EX260 SPN操作手册指令配置参数以控制每个点位 去SMC官网下载EX260 GSD文件及EX260 SPN operation manual操作手册 https www smc eu
  • 谈一谈浏览器与Node.js中的JavaScript事件循环,宏任务与微任务机制

    JavaScript中的异步代码 JavaScript是一个单线程非阻塞的脚本语言 这代表代码是执行在一个主线程上面的 但是JavaScript中有很多耗时的异步操作 例如AJAX setTimeout等等 也有很多事件 例如用户触发的点击