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先查找同步任务,然后再去任务队列中查找微任务,最后才去任务队列中查找宏任务;
  • 执行顺序:同步任务 》微任务 》宏任务。
相关推荐
X 西安7 分钟前
第十章JavaScript的应用
开发语言·javascript·ecmascript
想你的风吹到了瑞士13 分钟前
变量提升&函数提升
前端·javascript·vue.js
夫琅禾费米线1 小时前
leetcode2650. 设计可取消函数 generator和Promise
开发语言·javascript·leetcode·ecmascript
战族狼魂2 小时前
html+js实现图片的放大缩小等比缩放翻转,自动播放切换,顺逆时针旋转
javascript·css·html
Komorebi⁼2 小时前
Vue核心特性解析(内含实践项目:设置购物车)
前端·javascript·vue.js·html·html5
明月清风徐徐2 小时前
Vue实训---0-完成Vue开发环境的搭建
前端·javascript·vue.js
MR·Feng2 小时前
使用Electron将vue2项目打包为桌面exe安装包
前端·javascript·electron
萧大侠jdeps3 小时前
图片生成视频-右进
前端·javascript·音视频
Domain-zhuo3 小时前
JS对于数组去重都有哪些方法?
开发语言·前端·javascript
明月清风徐徐3 小时前
Vue实训---2-路由搭建
前端·javascript·vue.js