JavaScript中的宏任务和微任务

面试中经常会被问到什么宏任务和微任务?工作中也会出一个奇怪的问题,两行代码,一会A结果现出来,一会B结果先出来,搞得一头雾水。有些人为了懒省事,全都是用async await,亦或者写个setTimeout,也不去追究原理,能解决问题就行。接下来就对这个问题展开讨论。

什么是宏任务和微任务?

宏任务是由宿主(浏览器、Node)发起的,一般在JavaScript引擎空闲时执行(不知道这样说合不合理)。常见的宏任务有:setTimeoutsetIntervalAjaxDOM事件

微任务是指在当前任务执行结束后立即执行的任务。常见的微任务有:Promiseasync/await。其中Promise本身是同步的,.then/.catch才是异步的。

执行顺序

首先,在JavaScript中,任务分为同步任务和异步任务,异步任务又有宏任务和微任务。同步任务在主线程中,异步任务会进入任务队列。

同步任务 》微任务 》宏任务

一起来看一下下面这个例子:

js 复制代码
console.log(1);
setTimeout(() => {
  console.log("setTimeout");
}, 1000);
new Promise((resolve, reject) => {
  console.log("Promise1");
  resolve("resolve");
  console.log("Promise2");
}).then((data) => {
  console.log(data);
});
console.log(2);

先区分一下都是什么任务:

按照上面的顺序,依次输出1Promise1Promise22resolvesetTimeout,这里值得说一下Promise本身是同步的,.then/.catch才是异步的,所以会先输出Promise1,而resolve("resolve")是告诉Promise成功了,并不阻碍进程,所以会继续输出Promise2

可能有人会说setTimeout一秒后才执行,肯定最后输出了,再看一个例子,把setTimeout设置为0,应该会立即执行吧?:

js 复制代码
console.log(1);
setTimeout(() => {
  console.log("setTimeout");
}, 1000);
setTimeout(() => {
  console.log("setTimeout0");
}, 0);
new Promise((resolve, reject) => {
  console.log("Promise1");
  resolve("resolve");
  console.log("Promise2");
}).then((data) => {
  console.log(data);
});
console.log(2);

也是先分析都是什么任务,依次输出1Promise1Promise22resolvesetTimeout0setTimeout,虽然setTimeout设置为0,他也是宏任务,按照执行顺序,也是要排队的。

再看一个例子:

js 复制代码
console.log(1);
setTimeout(() => {
  console.log("setTimeout");
}, 10);
console.log(2);
for (let index = 0; index < 10000; index++) {
  console.log(10000);
}

定时器设置10毫秒,下面循环10000次,肯定会超过10毫秒了,应该会先输出setTimeout了吧,实时并不是这样,因为for是同步的,但是JavaScript引擎一直在忙没空去搭理它。

总结

  • JavaScript中的任务分为同步任务和异步任务,异步任务又有宏任务和微任务;
  • JavaScript先查找同步任务,然后再去任务队列中查找微任务,最后才去任务队列中查找宏任务;
  • 执行顺序:同步任务 》微任务 》宏任务。
相关推荐
一颗花生米。15 分钟前
深入理解JavaScript 的原型继承
java·开发语言·javascript·原型模式
学习使我快乐0119 分钟前
JS进阶 3——深入面向对象、原型
开发语言·前端·javascript
bobostudio199520 分钟前
TypeScript 设计模式之【策略模式】
前端·javascript·设计模式·typescript·策略模式
勿语&1 小时前
Element-UI Plus 暗黑主题切换及自定义主题色
开发语言·javascript·ui
一路向前的月光6 小时前
Vue2中的监听和计算属性的区别
前端·javascript·vue.js
长路 ㅤ   6 小时前
vue-live2d看板娘集成方案设计使用教程
前端·javascript·vue.js·live2d
Fan_web6 小时前
jQuery——事件委托
开发语言·前端·javascript·css·jquery
Jiaberrr7 小时前
Element UI教程:如何将Radio单选框的圆框改为方框
前端·javascript·vue.js·ui·elementui
安冬的码畜日常9 小时前
【D3.js in Action 3 精译_029】3.5 给 D3 条形图加注图表标签(上)
开发语言·前端·javascript·信息可视化·数据可视化·d3.js
太阳花ˉ9 小时前
html+css+js实现step进度条效果
javascript·css·html