macrotask与microtask
在说明宏任务及微任务前总结一下事件循环机制(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的更多相关文章
- javascript中的异步 macrotask 和 microtask 简介
javascript中的异步 macrotask 和 microtask 简介 什么是macrotask?什么是microtask?在理解什么是macrotask?什么是microtask之前,我们先 ...
- 异步 JavaScript 之 macrotask、microtask
1.异步任务运行机制 先运行下面的一段代码: console.log('script start'); setTimeout(function() { console.log('setTimeout' ...
- 事件循环进阶:macrotask与microtask
这段参考了参考来源中的第2篇文章(英文版的),(加了下自己的理解重新描述了下), 这里没法给大家演示代码,我就简单说下我的理解吧. promise和settimeout 在一起的时候执行顺序是个有意思 ...
- 异步 JavaScript 之理解 macrotask 和 microtask(转)
这个知识点... https://blog.keifergu.me/2017/03/23/difference-between-javascript-macrotask-and-microtask/? ...
- JavaScript event loop事件循环 macrotask与microtask
macrotask 姑且称为宏任务,在很多上下文也被简称为task.例如: setTimeout, setInterval, setImmediate, I/O, UI rendering. mic ...
- JavaScript中的异步 macrotask 和 microtask
看过很多setTimeout.Promise执行顺序的面试题,一直不明白为啥都是异步操作,Promise就牛×些呢?直到了解了macrotask和micromask才恍然大悟... 先来一道面试题助助 ...
- 聊聊JavaScript异步中的macrotask和microtask
前言 首先来看一个JavaScript的代码片段: console.log(1); setTimeout(() => { console.log(2); Promise.resolve().th ...
- Javascript中的Microtask和Macrotask——从一道很少有人能答对的题目说起
首先我们来看一道题目,如下javascript代码,执行后会在控制台打印出什么内容? async function async1() { console.log('async1 start'); aw ...
- node的“宏任务(macro-task)”和“微任务(micro-task)”机制
macrotask 和 microtask 表示异步任务的两种分类.在挂起任务时,JS 引擎会将所有任务按照类别分到这两个队列中,首先在 macrotask 的队列(这个队列也被叫做 task que ...
随机推荐
- elasticsearch7 配置篇
学习了这么多,终于开始搭建生产环境了,这一篇主要讲解配置项,以及支持中文分词的ik安装,集群的搭建. 配置项确实挺多的,但把几个常用配置熟悉就好,而且就像elasticsearch官方文档所说,不存在 ...
- dos转unix
方式一 # yum install dos2unix.x86_64 # dos2unix file 方式二 查看样式: :set ff? //dos/unix 设置: :set fileformat= ...
- 如何在一个ubuntu系统上搭建SVN版本控制工具
有话说,由于公司项目部署需要,将Windows工程迁移到Linux,通过调查确定使用Ubuntu的Linux操作系统.那么如何快速搭建和Windows一样快捷方便的开发环境就很重要了.本文讲述如何在一 ...
- Druid连接池使用
转载请注明原文地址:https://www.cnblogs.com/ygj0930/p/11280540.html 一:DRUID连接池简介 阿里出品的“为监控而生”的数据库连接池,在功能.性能.扩展 ...
- Tests in error:BlogApplicationTests.initializationError » IllegalState Unable to find a @Spri...【解决】
刚刚写完一个项目,准备打包,却发现无法打包. 然后认真排查了一下问题.发现少引入了一个插件. <plugin> <groupId>org.apache.maven.plugin ...
- 关于苹果macOS更新到Catalina后出现的各种问题(持续更新)
一.Mac系统更新后 Git 不能用,提示 missing xcrun at:xxx xcrun: error: invalid active developer path (/Library/Dev ...
- swarm 集群
1.创建manage节点 $docker swarm init --advertise-addr=192.168.0.38(主机ip) To add a worker to this swarm, r ...
- Python进阶-Ⅷ 匿名函数 lambda
1.匿名函数的引入 为了解决那些功能很简单的需求而设计的一句话函数 def func(i): return 2*i # 简化之后 func = lambda i:2*i #todo 其中:func是函 ...
- Excel 文本函数
1.FIND函数--要查找的字符在 字符串中 的 位置 FIND(find_text,within_text,start_num) Find_text 是要查找的字符串. Within_text 是 ...
- Apex 中插入更新数据的事件执行顺序
在使用 Apex 代码插入或更新数据的时候,若干事件会被按顺序执行.了解这些顺序可以提高调试程序的效率,也可以避免不必要的错误. 可以参考官方文档. 事件的执行顺序 从数据库中读取要更新的数据记录或初 ...