在说明宏任务及微任务前总结一下事件循环机制(event loop):
  • 首先判断JS是同步还是异步,同步就进入主线程,异步就进入event table
  • 异步任务在event table中注册函数,当满足触发条件后,被推入event queue
  • 同步任务进入主线程后一直执行,直到主线程空闲时,才会去event queue中查看是否有可执行的异步任务,如果有就推入主线程中

macrotask(宏任务):主代码块、setTimeout、setInterval等

microtask(微任务):Promise、process.nextTick等

下面通过【今日头条】这道面试题详细说明一下:
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' )
先画一个通用任务表(后面的步骤就是给表插值的过程,执行顺序是从上至下,从左往右,先同步后异步)
宏任务1 宏任务2 宏任务3 ... 宏任务n
微任务1.1 微任务2.1 微任务3.1 ... 微任务n.1
微任务1.2 微任务2.2 微任务3.2 ... 微任务n.2
微任务1.3 微任务2.3 微任务3.3 ... 微任务n.3
... ... ... ... ...
步骤详解:

1、咱们整道面试题就是宏任务1,async1()及async2()函数声明先不管,直接看第一步执行的代码:console.log( 'script start' ),没有疑问,同步微任务。

面试题主代码块 宏任务2 宏任务3 ... 宏任务n
console.log( 'script start' ) 微任务2.1 微任务3.1 ... 微任务n.1
微任务1.2 微任务2.2 微任务3.2 ... 微任务n.2
微任务1.3 微任务2.3 微任务3.3 ... 微任务n.3
... ... ... ... ...

2、然后到setTimeout,一个新的宏任务,即宏任务2,其中函数执行代码console.log( 'setTimeout' )及该宏任务中的微任务。

面试题主代码块 setTimeout 宏任务3 ... 宏任务n
console.log( 'script start' ) console.log( 'setTimeout' ) 微任务3.1 ... 微任务n.1
微任务1.2 微任务2.2 微任务3.2 ... 微任务n.2
微任务1.3 微任务2.3 微任务3.3 ... 微任务n.3
... ... ... ... ...

3、接下来是函数调用async1()console.log( 'async1 start' )同步代码进入宏任务1。

面试题主代码块 setTimeout 宏任务3 ... 宏任务n
console.log( 'script start' ) console.log( 'setTimeout' ) 微任务3.1 ... 微任务n.1
console.log( 'async1 start' ) 微任务2.2 微任务3.2 ... 微任务n.2
微任务1.3 微任务2.3 微任务3.3 ... 微任务n.3
... ... ... ... ...

4、随后执行await async2()语句,这里会先调用async2()中的同步代码,然后阻塞async1()的执行(直到该宏任务中所有同步代码执行完毕后再继续),具体原因可以参见我之前写的:async与await总结,所以async2()中同步代码console.log( 'async2' )进入宏任务1。

所以这里先把宏任务1中待执行的异步代码console.log( 'async1 end' )放最后面(其实是事件循环机制会在同步代码执行完毕即主线程空闲时,遍历一遍异步微任务,看是否有可执行的异步任务,有的话就推入主线程,我们人工怕遗留,所以先放后面):

面试题主代码块 setTimeout 宏任务3 ... 宏任务n
console.log( 'script start' ) console.log( 'setTimeout' ) 微任务3.1 ... 微任务n.1
console.log( 'async1 start' ) 微任务2.2 微任务3.2 ... 微任务n.2
console.log( 'async2' ) 微任务2.3 微任务3.3 ... 微任务n.3
... ... ... ... ...
console.log( 'async1 end' ) ... ... ... ...

5、下面是执行new Promise(),Promise构造函数是直接调用的同步代码,所以将console.log( 'promise1' )加入宏任务1,但其中resolve()的执行为异步代码(实际为promise.then()的执行),所以将console.log( 'promise2' )依次放最后面:

面试题主代码块 setTimeout 宏任务3 ... 宏任务n
console.log( 'script start' ) console.log( 'setTimeout' ) 微任务3.1 ... 微任务n.1
console.log( 'async1 start' ) 微任务2.2 微任务3.2 ... 微任务n.2
console.log( 'async2' ) 微任务2.3 微任务3.3 ... 微任务n.3
console.log( 'promise1' ) ) 微任务2.4 微任务3.4 ... 微任务n.4
... ... ... ... ...
console.log( 'async1 end' ) ... ... ... ...
console.log( 'promise2' ) ... ... ... ...

6、最后是console.log( 'script end' )同步代码的执行,放入宏任务1,同步之后,异步之前。

面试题主代码块 setTimeout 宏任务3 ... 宏任务n
console.log( 'script start' ) console.log( 'setTimeout' ) 微任务3.1 ... 微任务n.1
console.log( 'async1 start' ) 微任务2.2 微任务3.2 ... 微任务n.2
console.log( 'async2' ) 微任务2.3 微任务3.3 ... 微任务n.3
console.log( 'promise1' ) ) 微任务2.4 微任务3.4 ... 微任务n.4
console.log( 'script end' ) ... ... ... ...
console.log( 'async1 end' ) ... ... ... ...
console.log( 'promise2' ) ... ... ... ...

7、最后按照从上至下,从左往右的顺序执行就可以得到最终的结果了。

最终的打印顺序:

script start

async1 start

async2

promise1

script end

async1 end

promise2

setTimeout

参考资料:

1、8张图看清 async/await 和 promise 的执行顺序(原文中async1 end和promise2弄反了,需要注意):https://segmentfault.com/a/1190000017224799?_ea=5345525

macrotask与microtask的更多相关文章

  1. javascript中的异步 macrotask 和 microtask 简介

    javascript中的异步 macrotask 和 microtask 简介 什么是macrotask?什么是microtask?在理解什么是macrotask?什么是microtask之前,我们先 ...

  2. 异步 JavaScript 之 macrotask、microtask

    1.异步任务运行机制 先运行下面的一段代码: console.log('script start'); setTimeout(function() { console.log('setTimeout' ...

  3. 事件循环进阶:macrotask与microtask

    这段参考了参考来源中的第2篇文章(英文版的),(加了下自己的理解重新描述了下), 这里没法给大家演示代码,我就简单说下我的理解吧. promise和settimeout 在一起的时候执行顺序是个有意思 ...

  4. 异步 JavaScript 之理解 macrotask 和 microtask(转)

    这个知识点... https://blog.keifergu.me/2017/03/23/difference-between-javascript-macrotask-and-microtask/? ...

  5. JavaScript event loop事件循环 macrotask与microtask

    macrotask  姑且称为宏任务,在很多上下文也被简称为task.例如: setTimeout, setInterval, setImmediate, I/O, UI rendering. mic ...

  6. JavaScript中的异步 macrotask 和 microtask

    看过很多setTimeout.Promise执行顺序的面试题,一直不明白为啥都是异步操作,Promise就牛×些呢?直到了解了macrotask和micromask才恍然大悟... 先来一道面试题助助 ...

  7. 聊聊JavaScript异步中的macrotask和microtask

    前言 首先来看一个JavaScript的代码片段: console.log(1); setTimeout(() => { console.log(2); Promise.resolve().th ...

  8. Javascript中的Microtask和Macrotask——从一道很少有人能答对的题目说起

    首先我们来看一道题目,如下javascript代码,执行后会在控制台打印出什么内容? async function async1() { console.log('async1 start'); aw ...

  9. node的“宏任务(macro-task)”和“微任务(micro-task)”机制

    macrotask 和 microtask 表示异步任务的两种分类.在挂起任务时,JS 引擎会将所有任务按照类别分到这两个队列中,首先在 macrotask 的队列(这个队列也被叫做 task que ...

随机推荐

  1. 技能篇丨FineCMS 5.0.10 多个漏洞详细分析

    今天是一篇关于技能提升的文章,文章中的CMS是FineCMS,版本是5.0.10版本的几个漏洞分析,主要内容是介绍漏洞修补前和修补后的分析过程,帮助大家快速掌握该技能. 注:篇幅较长,阅读用时约7分钟 ...

  2. maven 学习---Maven构建自动化-Hudson

    建立自动化定义场景,依赖项目建设过程中被启动,一旦项目生成成功完成,以确保相关的项目是稳定的. 实例 考虑一个团队正在开发一个项目总线核心API上的其他两个项目的应用程序,网页UI和应用程序的桌面UI ...

  3. Centos7部署ejforum论坛(Java+tomcat+mysql)

    前面搭建Java环境和tomcat环境. 下面进行实战,搭建ejforum论坛 ejforum论坛源码:https://www.lanzous.com/i45rcoh Centos7安装MySQL数据 ...

  4. Java同步和异步,阻塞和非阻塞

    同步和异步.阻塞和非阻塞 同步和异步关注的是消息通信机制. 同步是指: 发送方发出数据后, 等待接收方发回响应后才发下一个数据包的通讯方式. 就是在发出一个调用时, 在没有得到结果之前, 该调用就不返 ...

  5. c# 第10节 表达式

    本节内容: 1:表达式是什么 2:表达式实例 1:表达式是什么 2:表达式实例 3:运算符分类

  6. python批量修改文件名(以修改图片名为例)

    #coding=utf-8import sys, string, os, shutil #输入目录名和前缀名,重命名后的名称结构类似prefix_0001 def RenameFiles(srcdir ...

  7. JDOJ 1139 水仙花数

    JDOJ 1139: 水仙花数 https://neooj.com/oldoj/problem.php?id=1139 Description 打印出所有"水仙花数",所谓&quo ...

  8. Fedora增加rc-local服务开机自启项

      最近新装了一台Fedora 30系统,服务已经正常运行起来了,但是偶然发现当我的系统重启后,写在rc.local配置文件里的命令居然没生效,导致我系统重启,但是服务却没有正常运行,后来经过一番查阅 ...

  9. B1001 害死人不偿命的(3n+1)猜想 (15 分)

    一.参考代码: #include<iostream> using namespace std; int main(){ int n; int step = 0; cin >> ...

  10. Graphviz学习

    (入门教程)[https://www.luogu.com.cn/blog/umr/graphviz]