(译)看得见的 JavaScript:事件循环(Event Loop)

简介: (译)看得见的 JavaScript:事件循环(Event Loop)

事件循环!这是每一个JS开发者都会遇到的东西,但一开始理解起来会很复杂。

首先,什么是事件循环,为什么你应该关注它?

JS 是单线程(single-threaded)的:一次只能运行一个任务。通常这没什么大问题,但是现在想象一下,你正在运行一个要耗时30s的任务……我们必须等着这30s过去才能执行其他的代码(JS默认在浏览器的主线程上运行,所以整个UI的解析会卡住。),都2019年了,没有人想要一个缓慢、没有响应的网页。

还好浏览器给了我们一些JS引擎没有提供的特性:Web API。它包括 DOM API,setTimeout,HTTP requests 等等。它能帮我们创建一些异步的、不会阻塞的行为。

当我们调用一个函数时,它将被添加到一个叫做调用栈的地方。调用栈是JS引擎的一部分,它不是浏览器的所特有的。这是一个堆栈,意味着先进后出(想象一堆煎饼堆叠在一起)。当一个函数返回了一个值时,它就会从调用栈中弹出。

1688268966361.png

respond函数返回一个 setTimeout函数,这个setTimeout就是 Web API 提供给我们的,它的作用是让我们延迟任务的执行而不是阻塞主线程。我们传给setTimeout的回调函数:一个箭头函数() => { return 'Hey' } 被加入到 Web API 中。同时,setTimeout函数和respond函数从调用栈中弹出,它们都返回了各自的值!

1688268994650.png

在 Web API 中,一个 1000ms 的定时器开始运行。这个回调函数不会立即被添加到调用栈,而是被传递到一个被称为队列的地方。


1688269007743.png


这是令人疑惑的部分:这并不意味着在1000ms之后,回调函数会被添加到调用栈中!它只是在1000ms后简单地添加到了队列中。正因为它是一个队列,所以函数必须在这里排队等待,等到轮到它了才会被放进调用栈中去执行。

现在是我们期待已久的时刻了——连接调用栈与队列!如果调用栈是空的,也就是说之前所有被调用的函数都返回了值并且弹出了调用栈,那么队列中的第一个任务将会被添加到调用栈中。在本例中,没有其他的函数被调用,这就说明当回调函数是队列中的第一项时,调用栈是空的。

1688269023822.png

回调函数添加到调用栈中,执行,然后返回一个值,最终弹出调用栈。

1688269037959.png


阅读一篇文章非常轻松有趣,但是你只有一遍又一遍地使用它才能完全掌握它。如果我们运行以下示例,想想看控制台里会出现什么?

const foo = () => console.log("First");
const bar = () => setTimeout(() => console.log("Second"), 500);
const baz = () => console.log("Third");
bar();
foo();
baz();

理解了吗?让我们快速地看一下当我们在浏览器运行这段代码时正在发生什么:

1688269063743.png


  1. 我们调用barbar返回一个setTimeout函数。
  2. 我们传给setTimeout的回调函数被添加到 Web API 中,然后 setTimeoutbar 都从调用栈中弹出。 3. 定时器运行,与此同时,foo 被调用,输出"First"foo返回undefinedbaz被调用,然后 Web API 中的回调函数被加入到队列中。
  3. baz输出"Third"。在baz执行完毕后,事件循环发现调用栈是空的,此时队列中的回调函数被添加到调用栈中。
  4. 回调函数输出"Second"

参考:dev.to/lydiahallie…

添加我的微信:enjoy_Mr_cat,共同成长,卷卷群里等你 🤪。

以上,感谢您的阅读~


目录
相关文章
|
存储 JavaScript 前端开发
深入理解JavaScript中的事件循环(Event Loop):机制与实现
【10月更文挑战第12天】深入理解JavaScript中的事件循环(Event Loop):机制与实现
469 3
|
8月前
|
消息中间件 JavaScript 前端开发
最细最有条理解析:事件循环(消息循环)是什么?为什么JS需要异步
度一教育的袁进老师谈到他的理解:单线程是异步产生的原因,事件循环是异步的实现方式。 本质是因为渲染进程因为计算机图形学的限制,只能是单线程。所以需要“异步”这个技术思想来解决页面阻塞的问题,而“事件循环”是实现“异步”这个技术思想的最主要的技术手段。 但事件循环并不是全部的技术手段,比如Promise,虽然受事件循环管理,但是如果没有事件循环,单一Promise依然能实现异步不是吗? 博客不应该只有代码和解决方案,重点应该在于给出解决方案的同时分享思维模式,只有思维才能可持续地解决问题,只有思维才是真正值得学习和分享的核心要素。如果这篇博客能给您
|
JavaScript 前端开发 开发者
JavaScript的事件循环
【10月更文挑战第27天】理解JavaScript的事件循环机制对于正确编写和理解JavaScript中的异步代码至关重要,它是JavaScript能够高效处理各种异步任务的关键所在。
241 59
|
12月前
|
JavaScript 前端开发 API
深入理解Node.js事件循环及其在后端开发中的应用
本文旨在揭示Node.js的核心特性之一——事件循环,并探讨其对后端开发实践的深远影响。通过剖析事件循环的工作原理和关键组件,我们不仅能够更好地理解Node.js的非阻塞I/O模型,还能学会如何优化我们的后端应用以提高性能和响应能力。文章将结合实例分析事件循环在处理大量并发请求时的优势,以及如何避免常见的编程陷阱,从而为读者提供从理论到实践的全面指导。
|
JavaScript API 开发者
深入理解Node.js中的事件循环和异步编程
【10月更文挑战第41天】本文将通过浅显易懂的语言,带领读者探索Node.js背后的核心机制之一——事件循环。我们将从一个简单的故事开始,逐步揭示事件循环的奥秘,并通过实际代码示例展示如何在Node.js中利用这一特性进行高效的异步编程。无论你是初学者还是有经验的开发者,这篇文章都能让你对Node.js有更深刻的认识。
|
前端开发 JavaScript
深入理解JavaScript中的事件循环(Event Loop):从原理到实践
【10月更文挑战第12天】 深入理解JavaScript中的事件循环(Event Loop):从原理到实践
438 1
|
前端开发 JavaScript UED
深入理解JavaScript中的事件循环机制
JavaScript中的事件循环机制是其异步编程的核心,深入理解该机制对于开发高效、流畅的前端应用至关重要。本文将介绍事件循环的工作原理、常见的事件循环模型,以及如何利用这些知识解决前端开发中的常见问题。
|
开发框架 JavaScript 前端开发
JavaScript的事件循环机制是其非阻塞I/O的关键
【5月更文挑战第13天】JavaScript的事件循环机制是其非阻塞I/O的关键,由调用栈、事件队列和Web APIs构成。当异步操作完成,回调函数进入事件队列,待调用栈空时,事件循环取队列中的任务执行。在游戏开发中,事件循环驱动游戏循环更新,包括输入处理、游戏逻辑更新和渲染。示例代码展示了如何模拟游戏循环,实际开发中则常使用游戏框架进行抽象处理。
245 4
|
前端开发 JavaScript UED
JavaScript 的事件循环机制是其非阻塞 I/O 模型的核心
【5月更文挑战第9天】JavaScript的事件循环机制是其非阻塞I/O的关键,通过单线程的调用栈和任务队列处理异步任务。当调用栈空时,事件循环从任务队列取出一个任务执行,形成循环。异步操作完成后,回调函数进入任务队列,等待被事件循环处理。微任务如Promise回调在每个宏任务结束后执行。此机制确保JavaScript能高效处理异步操作,不阻塞主线程,提供流畅的用户体验。
165 2
|
消息中间件 存储 前端开发
理解JavaScript事件循环机制
理解JavaScript事件循环机制
118 1

热门文章

最新文章

下一篇
oss云网关配置