JavaScript 事件循环机制

简介: javascript是一门单线程的非阻塞的脚本语言。单线程意味着javascript在执行代码的任何时候,都只有一个主线程来处理所有的任务。

javascript是一门单线程的非阻塞的脚本语言。单线程意味着javascript在执行代码的任何时候,都只有一个主线程来处理所有的任务。


那么javascript引擎是如何实现这一点的呢?


因为事件循环(event loop)。先上图:


image.png


图片解读:


  • 同步和异步任务分别进入不同的执行场所,同步的进入主线程,异步的进入Event Table并注册函数


  • 当指定的事情完成时(重点)Event Table会将这个函数移入Event Queue


  • 主线程内的任务执行完毕为空,会去Event Queue读取对应的函数,进入主线程执行


  • 上述的过程会不断的重复,也就是常常说的Event Loop(事件循环)


简单例子


我们来一个简单的例子来说明下:


console.log('1');
setTimeout(() => {
  console.log('2');
}, 0)
console.log('3');


上面的代码将输出下面的结果:


1
3
2


因为setTimeout是一个异步的任务,所以会在最后才执行。


那么,我们来个复杂点的例子:


复杂例子


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


上面的代码输出的结果是:


1
4
5
7
6
3
2


看到这代码的时候是不是有些蒙圈?在我们揭开谜底之前,先来了解下微任务和宏任务


微任务和宏任务


微任务和宏任务都是异步的任务,他们都属于队列,主要区别是它们的执行顺序--微任务会比宏任务先执行。


宏任务包含有:setTimeout, setInterval, setImmediate, I/O, UI rendering


微任务包含有:process.nextTick, promise.then, MutationObserver


嗯~回到上面的代码,如下:


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


在执行到new Promise的时候会立马新建一个promise对象并立即执行。所以会输出 1,4,5,而then则会在Event Table中注册成回调函数并放在微任务队列中,而两个setTimeout(输出3)和setTimeout(输出2,1s后完成的啊)会被先后注册成回调函数并放在宏任务队列中。


理解了上面的一些原理之后,我们再来练下手...


console.log(1)
process.nextTick(() => {
  console.log(8)
  setTimeout(() => {
    console.log(9)
  })
})
setTimeout(() => {
  console.log(2)
  new Promise(() => {
    console.log(11)
  })
})
let promise = new Promise((resolve,reject) => {
  setTimeout(() => {
    console.log(10)
  })
  resolve()
  console.log(4)
})
fn()
console.log(3)
promise.then(() => {
  console.log(12)
})
function fn(){
  console.log(6)
}


得到的结果是:


1
4
6
3
8
12
2
11
10
9


客官可以画下图整理下思路,然后代码运行验证一下啊💨


相关文章
|
2月前
|
存储 JavaScript 前端开发
深入理解JavaScript中的事件循环(Event Loop):机制与实现
【10月更文挑战第12天】深入理解JavaScript中的事件循环(Event Loop):机制与实现
125 3
|
1月前
|
JavaScript 前端开发 API
深入理解Node.js事件循环及其在后端开发中的应用
本文旨在揭示Node.js的核心特性之一——事件循环,并探讨其对后端开发实践的深远影响。通过剖析事件循环的工作原理和关键组件,我们不仅能够更好地理解Node.js的非阻塞I/O模型,还能学会如何优化我们的后端应用以提高性能和响应能力。文章将结合实例分析事件循环在处理大量并发请求时的优势,以及如何避免常见的编程陷阱,从而为读者提供从理论到实践的全面指导。
|
1月前
|
JavaScript API 开发者
深入理解Node.js中的事件循环和异步编程
【10月更文挑战第41天】本文将通过浅显易懂的语言,带领读者探索Node.js背后的核心机制之一——事件循环。我们将从一个简单的故事开始,逐步揭示事件循环的奥秘,并通过实际代码示例展示如何在Node.js中利用这一特性进行高效的异步编程。无论你是初学者还是有经验的开发者,这篇文章都能让你对Node.js有更深刻的认识。
|
3月前
|
JavaScript 安全 前端开发
乾坤js隔离机制
乾坤js隔离机制
|
1月前
|
JavaScript 安全 中间件
深入浅出Node.js中间件机制
【10月更文挑战第36天】在探索Node.js的奥秘之旅中,中间件的概念如同魔法一般,它让复杂的请求处理变得优雅而高效。本文将带你领略这一机制的魅力,从概念到实践,一步步揭示如何利用中间件简化和增强你的应用。
|
1月前
|
JavaScript 前端开发 开发者
JavaScript的事件循环
【10月更文挑战第27天】理解JavaScript的事件循环机制对于正确编写和理解JavaScript中的异步代码至关重要,它是JavaScript能够高效处理各种异步任务的关键所在。
40 1
|
1月前
|
消息中间件 JavaScript 中间件
深入浅出Node.js中间件机制
【10月更文挑战第24天】在Node.js的世界里,中间件如同厨房中的调料,为后端服务增添风味。本文将带你走进Node.js的中间件机制,从基础概念到实际应用,一探究竟。通过生动的比喻和直观的代码示例,我们将一起解锁中间件的奥秘,让你轻松成为后端料理高手。
37 1
|
1月前
|
Web App开发 JSON JavaScript
Node.js 中的中间件机制与 Express 应用
Node.js 中的中间件机制与 Express 应用
|
2月前
|
前端开发 JavaScript
深入理解JavaScript中的事件循环(Event Loop):从原理到实践
【10月更文挑战第12天】 深入理解JavaScript中的事件循环(Event Loop):从原理到实践
45 1
|
2月前
|
Web App开发 JavaScript 前端开发
深入理解Node.js事件循环和异步编程模型
【10月更文挑战第9天】在JavaScript和Node.js中,事件循环和异步编程是实现高性能并发处理的基石。本文通过浅显易懂的语言和实际代码示例,带你一探究竟,了解事件循环的工作原理及其对Node.js异步编程的影响。从基础概念到实际应用,我们将一步步解锁Node.js背后的魔法,让你的后端开发技能更上一层楼!