多图生动详解浏览器与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


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

相关文章
|
1月前
|
存储 JavaScript 网络协议
浏览器与 Node 的事件循环
浏览器和Node.js的事件循环是异步操作的核心机制。它们通过管理任务队列和回调函数,确保程序在处理耗时任务时不会阻塞主线程,从而实现高效、响应式的应用开发。
|
1月前
|
机器学习/深度学习 自然语言处理 前端开发
前端神经网络入门:Brain.js - 详细介绍和对比不同的实现 - CNN、RNN、DNN、FFNN -无需准备环境打开浏览器即可测试运行-支持WebGPU加速
本文介绍了如何使用 JavaScript 神经网络库 **Brain.js** 实现不同类型的神经网络,包括前馈神经网络(FFNN)、深度神经网络(DNN)和循环神经网络(RNN)。通过简单的示例和代码,帮助前端开发者快速入门并理解神经网络的基本概念。文章还对比了各类神经网络的特点和适用场景,并简要介绍了卷积神经网络(CNN)的替代方案。
|
6月前
|
前端开发
windows10 安装node npm 等前端环境 并配置国内源
windows10 安装node npm 等前端环境 并配置国内源
344 3
|
3月前
|
SQL JavaScript 数据库
sqlite在Windows环境下安装、使用、node.js连接
sqlite在Windows环境下安装、使用、node.js连接
|
3月前
|
JavaScript 前端开发 Windows
NodeJS的环境部署
介绍如何在Windows操作系统上安装Node.js环境,包括下载长期支持版本的Node.js、安装程序、编写测试代码并执行,以及如何在WebStorm集成开发环境中配置和运行Node.js。
50 1
|
4月前
|
编解码 JavaScript 前端开发
JS逆向浏览器脱环境专题:事件学习和编写、DOM和BOM结构、指纹验证排查、代理自吐环境通杀环境检测、脱环境框架、脱环境插件解决
JS逆向浏览器脱环境专题:事件学习和编写、DOM和BOM结构、指纹验证排查、代理自吐环境通杀环境检测、脱环境框架、脱环境插件解决
128 1
|
4月前
|
缓存 JavaScript Ubuntu
Node.js环境怎么搭建?
【8月更文挑战第4天】Node.js环境怎么搭建?
72 1
|
4月前
|
JavaScript Serverless Linux
函数计算产品使用问题之遇到Node.js环境下的请求日志没有正常输出时,该如何排查
函数计算产品作为一种事件驱动的全托管计算服务,让用户能够专注于业务逻辑的编写,而无需关心底层服务器的管理与运维。你可以有效地利用函数计算产品来支撑各类应用场景,从简单的数据处理到复杂的业务逻辑,实现快速、高效、低成本的云上部署与运维。以下是一些关于使用函数计算产品的合集和要点,帮助你更好地理解和应用这一服务。
|
4月前
|
JavaScript Linux API
【Azure 应用服务】NodeJS Express + MSAL 应用实现AAD集成登录并部署在App Service Linux环境中的实现步骤
【Azure 应用服务】NodeJS Express + MSAL 应用实现AAD集成登录并部署在App Service Linux环境中的实现步骤
|
4月前
|
JavaScript Linux 容器
【Azure 应用服务】NodeJS项目部署在App Service For Linux环境中,部署完成后应用无法访问
【Azure 应用服务】NodeJS项目部署在App Service For Linux环境中,部署完成后应用无法访问