多图生动详解浏览器与Node环境下的Event Loop(上)

简介: 今年秋招,在美团一面中被问到了这样一个问题:听过Event Loop吗?

前言


今年秋招,在美团一面中被问到了这样一个问题:听过Event Loop吗?


当时的我是一脸懵逼的,因为从来都没有听过这个专业名词。不过面试官还是很友好的,他说没关系,那你来做一道题,看看下面这段代码的执行结果是什么?


console.log('1')
setTimeout(function callback(){
 console.log('2')
}, 1000)
new Promise((resolve, reject) => {
    console.log('3')
    resolve()
})
.then(res => {
    console.log('4');
})
console.log('5')


很明显这道题考察的就是你对Event Loop的认识,不出所料,我当时就没做对这道题,但我默默得记下了这个考题,现在学习后回来整理知识点


你们知道正确答案是什么吗?这里先埋下个伏笔,大家可以自己做一做这道题,答案会在文章中公布


一、JavaScript是如何工作的


在刚开始学习JavaScript时,你一定听过这样一句话:JavaScript是单线程的


什么是单线程呢?就是很多段JS代码,它的执行顺序是从上到下一行一行执行的,即只有当上一行的代码执行完后才会执行下一行代码


这样的设定也是为了保证我们在实现某些功能时的代码逻辑的顺序性


此时有些人就会提出问题,上来就甩了一段代码给我,代码如下


console.log('1')
setTimeout(function (){
 console.log('2')
}, 1000)
console.log('3')
/* 运行结果:
 1
 3
 2
*/


不是说JS是单线程的,一行一行代码执行的吗?为什么这段代码先打印了 3 ,再打印了 2 呢?


先给出一个知识点,在JS中有些代码是异步执行的,所谓异步,就是不会阻塞代码的运行,而会另外开启一个空间去执行这段异步代码,其余同步的代码就仍正常执行,若异步代码中有其它的代码,则会在之后的某个时刻将异步代码中其它代码执行。例如上述代码中的 setTimeout 函数就是异步的,而其内部还有一段同步代码 console.log('2')


这里提到的某个时刻,也正是我们本文后续要讲到的重点,这里就先不做过多讲解


那么异步执行的额外空间是哪里来的?那当然是JS所处的运行环境提供的了,而JS最主要的两个运行环境就是:浏览器Node,我们接下来也会基于这两个运行环境,对JS的运行机制进行讲解


二、浏览器中的JavaScript


之所以JS能在浏览器中运行,那是因为浏览器都默认提供了一个JavaScript引擎,为JS提供一个运行环境


下图是一个JavaScript引擎的简化图:


53443dfa9a100ec55d82ef2e667548a5.png


图中左侧是内存堆heap,是浏览器为了给代码分配运行内存;图中右侧是调用栈stack,每当运行一段代码JS代码时,都会将代码压入调用栈中,然后在执行完毕以后出栈


对于内存堆我们就不做过多的了解,主要讲一下调用栈


(1)调用栈


什么是调用栈?这里有一段代码,我们通过它来分析一下调用栈的运行过程


function multiply(a, b) {
 return a * b
}
function calculate(n) {
 return multiply(n, n)
}
function print(n) {
 let res = calculate(n)
 console.log(res)
}
print(5)


当这段代码在浏览器中运行时,会先查询三个定义好了的函数 multiplycalculateprint ;然后执行 print(5) 这段代码,因为这三个函数是有调用关系的,因此接下来依次调用了 calculate 函数 、multiply 函数


现在,我们来看一下这段代码在执行过程中,调用栈stack内部的情况如何


f1d3c039f269892f5f11d108c70a7462.jpg


这里,还有一种方式可以来验证一下调用栈的存在以及其内容,我们来编写一段这样的代码:


function fn() {
    throw new Error('isErr')
}
function foo() {
    fn()
}
function main() {
    foo()
}
main()


然后在浏览器中运行一下,就会得到如下结果:


11fac0c4ec33b9fb8a3830beda8e298f.png


在代码运行过程中抛出错误时,浏览器将整个调用栈里的内容都打印了出来,正如我们所期望的一样,此时的调用栈是这个样子的:


cdf7aacc3764721386e3d62a372ee2c2.png


以上的过程涉及到的都是同步的代码,那么对于异步的代码来说,是如何像我们上面所说的一样,开辟一个新的空间去给异步代码运行的呢?


这里就要引入 Event Loop 的概念了


(2)Event Loop


Event Loop 翻译过来叫做事件循环,那到底是什么事件在循环呢?这里我们给出完整的浏览器的事件循环简图,来看一下


94bc6e3445ff8425034a728048cd627a.png


浏览器中的各种 Web API 为异步的代码提供了一个单独的运行空间,当异步的代码运行完毕以后,会将代码中的回调送入到 Task Queue(任务队列)中去,等到调用栈空时,再将队列中的回调函数压入调用栈中执行,等到栈空以及任务队列也为空时,调用栈仍然会不断检测任务队列中是否有代码需要执行,这一过程就是完整的Event Loop 了


我们可以用一个简单的例子,来感受一下事件循环的过程


console.log('1')
setTimeout(function callback(){
 console.log('2')
}, 1000)
console.log('3')


再通过动图来看看大致的过程


7dba181c5a9797f895a250e65168ea6d.jpg


在这里插入图片描述


(3)宏任务和微任务


简单理解了 Event Loop 的过程后,我们再来看一道题,看看是否能回答正确


console.log('1')
setTimeout(function callback(){
 console.log('2')
}, 1000)
new Promise((resolve, reject) => {
    console.log('3')
    resolve()
})
.then(res => {
    console.log('4');
})
console.log('5')
// 这段代码的打印结果顺序如何呢?


下面公布一下答案


// 正确答案:
   1
   3
   5
   4
   2


这里你是否又有个疑问了,为什么 promisesetTimeout 同样是异步,为什么前者优先于后者?


这里就要引入另外两个概念了,即 macrotask(宏任务) 和 microtask(微任务)

下面列举了我们浏览器中常用的宏任务和微任务


名称 举例(常用)
宏任务 setTimeout 、setInterval 、UI rendering
微任务 promise 、requestAnimationFrame


并且规定,当宏任务和微任务都处于 Task Queue 中时,微任务的优先级大于宏任务,即先将微任务执行完,再执行宏任务


因此,上述代码先打印了 4 ,再打印了 2


当然,既然区分了宏任务和微任务,那么存放它们的队列也就分为两种,分别为macro task queue(宏队列) 和 micro task queue(微队列),如图所示


13032bfcbbfe9462cf5ce5d26416c2eb.png


根据相关规定,当调用栈为空时,对于这两个队列的检测情况步骤如下:


  1. 检测微队列是否为空,若不为空,则取出一个微任务入栈执行,然后执行步骤1;若为空,则执行步骤2


  1. 检测宏队列是否为空,若不为空,则取出一个宏任务入栈执行,然后执行步骤1;若为空,直接执行步骤1


  1. ……往复循环


那么我们来看一下刚才那段代码的具体调用过程吧(由于wx对动图的限制,我不得不把动图分成两部分,还请大家耐心观看,十分抱歉)


7dde0c7b39092403715b9641b8656749.jpg


cb7e924250cf425383a087920b5631ba.jpg


看完这段执行过程,再去写一下上面那道题,看看能否写对呢?

相关文章
如何在不同的浏览器环境中确保时间戳转换的兼容性?
通过以上这些措施,可以在不同的浏览器环境中提高时间戳转换的兼容性,确保应用能够在各种浏览器中稳定运行,为用户提供良好的体验。
239 58
|
弹性计算 JavaScript 前端开发
一键安装!阿里云新功能部署Nodejs环境到ECS竟然如此简单!
Node.js 是一种高效的 JavaScript 运行环境,基于 Chrome V8 引擎,支持在服务器端运行 JavaScript 代码。本文介绍如何在阿里云上一键部署 Node.js 环境,无需繁琐配置,轻松上手。前提条件包括 ECS 实例运行中且操作系统为 CentOS、Ubuntu 等。功能特点为一键安装和稳定性好,支持常用 LTS 版本。安装步骤简单:登录阿里云控制台,选择扩展程序管理页面,安装 Node.js 扩展,选择实例和版本,等待创建完成并验证安装成功。通过阿里云的公共扩展,初学者和经验丰富的开发者都能快速进入开发状态,开启高效开发之旅。
|
9月前
|
人工智能 监控 JavaScript
HarmonyOS5云服务技术分享--ArkTS开发Node环境
本文详细讲解了在HarmonyOS(ArkTS API 9及以上)中使用云函数的开发技巧,结合Node.js和HTTP触发器,从零开始手把手教学。内容涵盖核心能力、开发流程(配置到部署)、高阶优化及常见问题解决,并提供实际应用场景示例。助你快速掌握Serverless开发,提升效率,探索跨端协作与AI集成等未来方向。
|
12月前
|
JavaScript 前端开发 数据可视化
【01】Cocos游戏开发引擎从0开发一款游戏-cocos环境搭建以及配置-Cocos Creator软件系统下载安装-node环境-优雅草卓伊凡
【01】Cocos游戏开发引擎从0开发一款游戏-cocos环境搭建以及配置-Cocos Creator软件系统下载安装-node环境-优雅草卓伊凡
746 2
【01】Cocos游戏开发引擎从0开发一款游戏-cocos环境搭建以及配置-Cocos Creator软件系统下载安装-node环境-优雅草卓伊凡
|
弹性计算 JavaScript 前端开发
一键安装!阿里云新功能部署Nodejs环境到ECS竟然如此简单!
一键安装!阿里云新功能部署Nodejs环境到ECS竟然如此简单!
一键安装!阿里云新功能部署Nodejs环境到ECS竟然如此简单!
|
11月前
|
JavaScript Ubuntu Linux
如何在阿里云的linux上搭建Node.js编程环境?
本指南介绍如何在阿里云Linux服务器(Ubuntu/CentOS)上搭建Node.js环境,包含两种安装方式:包管理器快速安装和NVM多版本管理。同时覆盖全局npm工具配置、应用部署示例(如Express服务)、PM2持久化运行、阿里云安全组设置及外部访问验证等步骤,助你完成开发与生产环境的搭建。
|
JavaScript 前端开发 数据处理
模板字符串和普通字符串在浏览器和 Node.js 中的性能表现是否一致?
综上所述,模板字符串和普通字符串在浏览器和 Node.js 中的性能表现既有相似之处,也有不同之处。在实际应用中,需要根据具体的场景和性能需求来选择使用哪种字符串处理方式,以达到最佳的性能和开发效率。
337 63
|
JavaScript
node环境之Error: Cannot find module ‘chalk’ 报错无法解决的问题—-网上说让你npm install chalk 基本是没有用的-优雅草央千澈解决方案
node环境之Error: Cannot find module ‘chalk’ 报错无法解决的问题—-网上说让你npm install chalk 基本是没有用的-优雅草央千澈解决方案
984 13
node环境之Error: Cannot find module ‘chalk’ 报错无法解决的问题—-网上说让你npm install chalk 基本是没有用的-优雅草央千澈解决方案
node环境之当我们遇到需要付费的依赖库@fortawesome/fontawesome-pro导致npm install无法进行怎么办-fontawesome-pro依赖库
node环境之当我们遇到需要付费的依赖库@fortawesome/fontawesome-pro导致npm install无法进行怎么办-fontawesome-pro依赖库
395 3
node环境之当我们遇到需要付费的依赖库@fortawesome/fontawesome-pro导致npm install无法进行怎么办-fontawesome-pro依赖库
|
机器学习/深度学习 JavaScript Cloud Native
Node.js作为一种快速、可扩展的服务器端运行时环境
Node.js作为一种快速、可扩展的服务器端运行时环境
248 8