一文带你彻底搞懂浏览器中的Event Loop

引言

对于从事前端的同学来说,应该对浏览器中的 Event Loop有所了解。这个了解的过程,可能是自己总结,也可能是通过网上的各种讲解。可能,对于前者而言能总结出 Event Loop,说明对它的概念已通透于心。但是,对于后者而言,很可能跌入理解的误区,因为网上有很多种 Event Loop的讲解的版本,各自有描述的重点,并不是能很好地表达 Event Loop的执行流程(我曾经也是这样理解…)。

所以,今天,我们就来详细地认识一下浏览器中的 Event Loop

一、Event Loop 定义


首先,我们来回忆一下,Event Loop的意义何在?众所周知,浏览器的 JS引擎线程为单线程。那么,为了防止代码阻塞的情况出现,所以设置了 Event Loop这种事件循环机制。

往大地讲,它是将一些耗时可能会很长的代码通过异步执行,例如 XMLHTTPRequest请求过程。往小地讲,对于这个异步的过程,通常被称作任务队列消息队列,即 JS引擎在解析 JS代码的时候,会将异步的代码运行结果(通常是回调)放在任务队列中,而任务队列又被分为 marco task(宏任务)和 micro task(微任务)(它们各自由相应的队列维护)。而在每次宏任务执行后(执行一个宏任务),都会询问微任务队列,然后按照 FIFO 先进先出的原则清空微任务队列

常见的宏任务有:

  • 主代码块(即 JS 引擎解析解析同步代码的过程)
  • 页面渲染
  • setTimeout
  • setInterval
  • setImmediate

常见的微任务有:

  • Promise.then
  • MutationObserver

PS:由于只是针对浏览器的 Event Loop,就不提 Node 的 process.nextick 之类的了~

通过,上面的概念性的理解,我想大家可能对 Event Loop如何执行还是有点迷糊,那么接下来,我们就重点通过一个 Demo来深入了解一下 Event Loop的运行机制。

二、Event Loop 运行机制


首先,我们先看一段简单的代码:

console.log(1) // {1}
Promise.resolve().then(() => { // {2}
	console.log(3) 
	Promise.resolve().then(() => { 
		console.log(4)
	})
})
setTimoue(() => { // {3}
	console.log(5) 
	Promise.resolve().then(() => { 
		console.log(6)
	})
})
console.log(2) // {4}

最终会打印输出 1 2 3 4 5 6,这是为什么呢?

1.首先,JS引擎先解析 {1} 时,由于是同步的,所以直接打印出 1

2.然后,解析 {2},由于 Promise.then是微任务(记作任务 A),所以会将它的运行结果放入微任务队列,那么此时任务队列为:
在这里插入图片描述
3.接下来,解析 {3},由于 setTimeout是宏任务(记作任务 B),所以会讲它的运行结果放入宏任务队列,那么此时任务队列为:
在这里插入图片描述
4.然后,再解析 {4},由于是同步的,所以直接打印出 2,此时任务队列不变。

5.此时主代码块执行完了(即宏任务执行完一次),此时去询问微任务队列,清空微任务队列(此时微任务队列刚好只有任务 A),执行完之后的任务队列为:
在这里插入图片描述
并且,此时需要注意的是,在任务 A 的执行过程中,不仅同步打印了 3,而且还生成了一个微任务,即这个微任务也会进入微任务队列,然后出队,即输出 4,这样微任务队列才被清空。

6.清空后微任务队列,会去询问宏任务队列,即执行任务 B(回调),同步打印 5,并且生成一个微任务(记作任务C),讲该微任务的运行结果放到微任务队列中,此时任务队列为:
在这里插入图片描述
7.同样地,在执行完一次宏任务后,会清空微任务队列,即执行任务 C,打印出 6。而自此,微任务队列和宏任务队列都为空,Event Loop就运行结束了。

总结


不知道大家最后弄清楚 Event Loop了没,记忆的过程中有很重要的两点,首先微任务和宏任务是基于队列进行管理的,它遵循 FIFO的原则。并且在每一次宏任务执行一次结束后,都会清空微任务队列。只要,大家明白这两点,然后结合上面的例子理解,我想浏览器中的 Event Loop对你来说应该就是小菜一碟。

Python面向对象编程(Object-Oriented Programming,简称OOP)是一种编程范式,它将数据和操作封装在对象,通过对象之间的交互实现程序的设计和开发。下面是一些关键概念,帮助你更好地理解Python面向对象编程。 1. 类(Class):类是对象的蓝图或模板,描述了对象的属性和行为。它定义了对象的特征和方法。例如,我们可以定义一个名为"Car"的类来表示汽车,其包含属性(如颜色、型号)和方法(如加速、刹车)。 2. 对象(Object):对象是类的实例,是具体的实体。通过实例化类,我们可以创建一个对象。例如,我们可以创建一个名为"my_car"的对象,它是基于"Car"类的实例。 3. 属性(Attribute):属性是对象的特征,用于描述对象的状态。每个对象都可以具有一组属性。例如,"Car"类的属性可以包括颜色、型号等。 4. 方法(Method):方法是对象的行为,用于定义对象的操作。每个对象都可以具有一组方法。例如,"Car"类的方法可以包括加速、刹车等。 5. 继承(Inheritance):继承是一种机制,允许我们创建一个新类(称为子类),从现有类(称为父类)继承属性和方法。子类可以扩展或修改父类的功能。继承可以实现代码重用和层次化设计。 6. 多态(Polymorphism):多态是一种特性,允许不同类的对象对同一方法做出不同的响应。多态提高了代码的灵活性和可扩展性。 7. 封装(Encapsulation):封装是一种将数据和操作封装在对象的机制,隐藏了对象的内部实现细节,只暴露必要的接口给外部使用。这样可以保护数据的安全性,提供了更好的模块化和代码复用性。 通过理解这些概念,你可以更好地掌握Python面向对象编程。在实践,你可以使用类来创建对象,操作对象的属性和调用对象的方法,通过继承和多态实现代码的灵活性和可扩展性,通过封装保护数据的安全性和提高代码的可维护性。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值