JS事件循环-微任务-宏任务(原理讲解+面试题分析)

 更新时间:2023年01月08日 15:00:48   作者:既白biu  
这篇文章主要介绍了JS事件循环-微任务-宏任务的原理,本文章含有面试题分析,不管是面试者还是想要学习相关内容的都可以很好的理解、掌握这部分内容,需要的朋友可以参考下

前言

JS代码在运行时,有两种运行环境。

一是在浏览器中,二是在node中。

由于JS线程是单线程,在运行JS代码时,可能会遇到比较耗时的操作,比如setTimeout,或者是发送网络请求等,又由于JS线程是单线程,如果在解析耗时的代码时候,停在了这里,那执行代码的性能将是比较低的。

为了解决此问题,在浏览器、node环境下,其实是有事件循环机制的。

浏览器的事件循环

浏览器的事件循环

JS线程执行代码时候,遇到比较耗时的操作时,将这些操作交给浏览器去处理,然后这些操作会根据不同的种类放进微任务队列或者宏任务队列,宏任务队列和微任务队列都不为空的时候,只有等微任务队列为空,即微任务队列里面的事件全部都执行完之后,才会再去让宏任务队列中的事件出栈,之后交由JS线程去处理,执行代码。

事件循环大概就是如图所示的流程:

浏览器的宏任务、微任务

其实,在浏览器拿到那些有些不能同步处理的事件的时候,有的会加入宏任务队列,有的会加入微任务队列,那么一般我们如何区分呢?

一般情况下:加入宏任务队列和微任务队列的事件如下:

宏任务队列(macrotask queue):ajax、setTimeout、setInterval、DOM监听、UI Rendering等

微任务队列(microtask queue):Promise的then回调、 Mutation Observer API、queueMicrotask()。

那么这些事件的执行顺序是怎么样子的呢?

首先,有一个原则,宏任务队列里面的事件,要执行的话,一定是在确保微任务队列为空的情况下,即微任务队列里面的事件全部执行完的情况。

其次,main script里面的内容是最先执行的。

由此,可以得到执行顺序为:main script > 微任务队列里面的事件 > 宏任务里面的事件。

面试题一

题目如下:

setTimeout(function () {
  console.log("setTimeout1");
  new Promise(function (resolve) {
    resolve();
  }).then(function () {
    new Promise(function (resolve) {
      resolve();
    }).then(function () {
      console.log("then4");
    });
    console.log("then2");
  });
});

new Promise(function (resolve) {
  console.log("promise1");
  resolve();
}).then(function () {
  console.log("then1");
});

setTimeout(function () {
  console.log("setTimeout2");
});

console.log(2);

queueMicrotask(() => {
  console.log("queueMicrotask1")
});

new Promise(function (resolve) {
  resolve();
}).then(function () {
  console.log("then3");
});

// promise1
// 2
// then1
// queueMicrotask1
// then3
// setTimeout1
// then2
// then4
// setTimeout2

分析如下:

在第一个事件循环里面,main script、宏任务、微任务里面的事件如下:

在判断加入宏任务队列还是微任务队列时候,遵循如下原则:

宏任务队列(macrotask queue):ajax、setTimeout、setInterval、DOM监听、UI Rendering等
微任务队列(microtask queue):Promise的then回调、 Mutation Observer
API、queueMicrotask()。

按照这个原则,第一轮事件循环里面的事件如下:

先执行main script、然后微任务队列里面的,最后是宏任务队列里面的

// promise1
// 2
// then1
// queueMicrotask1
// then3

之后执行setTimeout1的宏任务,此时第二轮事件循环里面的内容如下:

第二轮事件循环执行内容如下:

// setTimeout1
// then2
// then4
// setTimeout2

综上:最后执行结果为:

// promise1
// 2
// then1
// queueMicrotask1
// then3
// setTimeout1
// then2
// then4
// setTimeout2

面试题二

题目如下:

// async function bar() {
//   console.log("22222")
//   return new Promise((resolve) => {
//     resolve()
//   })
// }

// async function foo() {
//   console.log("111111")

//   await bar()

//   console.log("33333")
// }

// foo()
// console.log("444444")


async function async1 () {
  console.log('async1 start')
  await async2();
  console.log('async1 end')
}

async function async2 () {
  console.log('async2')
}

console.log('script start')

setTimeout(function () {
  console.log('setTimeout')
}, 0)
 
async1();
 
new Promise (function (resolve) {
  console.log('promise1')
  resolve();
}).then (function () {
  console.log('promise2')
})

console.log('script end')

// script start
// async1 start
// async2
// promise1
// script end
// async1 end
// promise2
// setTimeout

第一轮事件循环里面的事件如下:

然后按照顺序执行,最后结果如下:

// script start
// async1 start
// async2
// promise1
// script end
// async1 end
// promise2
// setTimeout

node的事件循环

node的事件循环

浏览器的事件循环是是根据HTML5定义的规范来实现的,不同的浏览器可能会有不同的实现,而Node中是由libuv实现的。

首先我们看一下node的架构图:

我们可以从图中大致看出,事件循环是在libuv中实现的,libuv主要维护的是一个事件循环(Event Loop)和 线程池(worker threads)。

libuv是一个多平台的专注于异步IO的库,它最初是为Node开发的,但是现在也被使用到Luvit、Julia、pyuv等其他地方;

EventLoop负责调用系统的一些其他操作:文件的IO、Network、child-processes等

由图可以看出,事件循环就像是一个桥梁,是连接着应用程序的JavaScript(左边部分)和系统调用(右边线程池部分)之间的通道:

无论是我们的文件IO、数据库、网络IO、定时器、子进程,在完成对应的操作后,都会将对应的结果和回调函数放到事件循环(任务队列)中;

事件循环会不断的从任务队列中取出对应的事件(回调函数)来执行;

但是一次完整的事件循环Tick分成很多个阶段:

  1. 定时器(Timers):本阶段执行已经被 setTimeout() 和 setInterval() 的调度回调函数。
  2. 待定回调(Pending Callback):对某些系统操作(如TCP错误类型)执行回调,比如TCP连接时接收到ECONNREFUSED。idle, prepare:仅系统内部使用。
  3. 轮询(Poll):检索新的 I/O 事件;执行与 I/O 相关的回调;
  4. 检测(check):setImmediate() 回调函数在这里执行。
  5. 关闭的回调函数:一些关闭的回调函数,如:socket.on(‘close’, …)

node的宏任务、微任务

node中也有微任务和宏任务,执行的原则和在浏览器中一样,是先执行微任务,然后再执行宏任务,但是对于宏任务来说,是按照上图从上到下的顺序执行的。

具体对应的常见事件的执行顺序如下;

在微任务队列中:

  • next tick queue:process.nextTick;
  • other queue:Promise的then回调、queueMicrotask;

(是按照从上往下的事件顺序执行)

在宏任务队列:

  • timer queue:setTimeout、setInterval;
  • poll queue:IO事件;
  • check queue:setImmediate;
  • close queue:close事件

(同样是按照从上往下的事件顺序执行)

所以,综上所述,在每一次事件循环的tick中,会按照如下顺序来执行代码:

next tick microtask queue;
other microtask queue;
timer queue;
poll queue;
check queue;
close queue

当然,main script 依旧是最先执行的,只有main script执行结束后,才会按照上述顺序来执行代码。

面试题一

async function async1() {
  console.log('async1 start')
  await async2()
  console.log('async1 end')
}

async function async2() {
  console.log('async2')
}

console.log('script start')

setTimeout(function () {
  console.log('setTimeout0')
}, 0)

setTimeout(function () {
  console.log('setTimeout2')
}, 300)

setImmediate(() => console.log('setImmediate'));

process.nextTick(() => console.log('nextTick1'));

async1();

process.nextTick(() => console.log('nextTick2'));

new Promise(function (resolve) {
  console.log('promise1')
  resolve();
  console.log('promise2')
}).then(function () {
  console.log('promise3')
})

console.log('script end')

// script start
// async1 start
// async2
// promise1
// promise2
// script end
// nexttick1
// nexttick2
// async1 end
// promise3
// settimetout0
// setImmediate
// setTimeout2

第一轮事件循环里面的事件如下:

按照顺序自左向右执行,3s后执行setTimeout2,

最后的结果是:

// script start
// async1 start
// async2
// promise1
// promise2
// script end
// nexttick1
// nexttick2
// async1 end
// promise3
// settimetout0
// setImmediate
// setTimeout2

到此这篇关于JS事件循环-微任务-宏任务(原理讲解+面试题分析)的文章就介绍到这了,更多相关循环-微任务-宏任务内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • js实现选项卡效果

    js实现选项卡效果

    这篇文章主要为大家详细介绍了js实现选项卡效果,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-03-03
  • JavaScritp添加url参数并将参数加入到url中及更改url参数的方法

    JavaScritp添加url参数并将参数加入到url中及更改url参数的方法

    这篇文章给大家介绍javascript添加url参数方法,将参数加入到url中,涉及到url添加参数的相关知识,关于js添加url参数感兴趣的朋友可以参考下本篇文章
    2015-10-10
  • 微信小程序之事件交互操作实例分析

    微信小程序之事件交互操作实例分析

    这篇文章主要介绍了微信小程序之事件交互操作,结合实例形式分析了微信小程序事件响应、交互及界面布局等相关操作技巧与注意事项,需要的朋友可以参考下
    2018-12-12
  • JS实现微信摇一摇原理解析

    JS实现微信摇一摇原理解析

    这篇文章主要为大家详细介绍了JS微信摇一摇的实现原理,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-07-07
  • JavaScript语法约定和程序调试原理解析

    JavaScript语法约定和程序调试原理解析

    这篇文章主要介绍了JavaScript语法约定和程序调试原理解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-11-11
  • JavaScript era库的使用详解

    JavaScript era库的使用详解

    这篇本文主要给大家介绍了JavaScript era库的使用,使用 ora 这个 JavaScript 库可以在命令行应用程序中提供漂亮的加载状态提示,本文详细介绍如何使用该库,并结合多个例子演示其功能,需要的朋友可以参考下
    2024-02-02
  • setInterval计时器不准的问题解决方法

    setInterval计时器不准的问题解决方法

    在js中如果打算使用setInterval进行倒数,计时等功能,往往是不准确的,针对这个问题,本文有个不错的解决方案
    2014-05-05
  • JavaScript中reduce()的5个基本用法示例

    JavaScript中reduce()的5个基本用法示例

    这篇文章主要给大家介绍了关于JavaScript中reduce()的5个基本用法示例,文中通过示例代码以及图文介绍的非常详细,对大家学习或者使用js具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
    2020-07-07
  • 仿google adsense颜色选择器代码,从中易广告联盟程序提取

    仿google adsense颜色选择器代码,从中易广告联盟程序提取

    仿google adsense颜色选择器代码,从中易广告联盟程序提取...
    2007-11-11
  • 深入理解JavaScript系列(27):设计模式之建造者模式详解

    深入理解JavaScript系列(27):设计模式之建造者模式详解

    这篇文章主要介绍了深入理解JavaScript系列(27):设计模式之建造者模式详解,建造者模式可以将一个复杂对象的构建与其表示相分离,使得同样的构建过程可以创建不同的表示,需要的朋友可以参考下
    2015-03-03

最新评论