如果你已经知道了js中存在宏任务和微任务,那么你一定已经了解过promise了。因为在js中promise是微任务的一个入口。

先来看一道题:

  1. setTimeout(function(){
  2. console.log('setTimeout')
  3. });
  4. new Promise(function(resolve, reject){
  5. console.log('pormise body');
  6. resolve();
  7. }).then(function(){
  8. console.log('promise then')
  9. });
  10. console.log('main');

这题的答案是:

  1. promise body
  2. main
  3. promise then
  4. setTimeout

promise body出现在第一行一点也不意外,意外的是,setTimeout出现在了promise then的后边。

 

setTimeout和promise then都是异步调用,setTimeout又定义在promise之前,如果没有意外,应该是先输出setTimeout才对,但这里却恰恰相反。

 

这里涉及到的知识点,就是宏任务与微任务了。标记一下上边的代码:

  1. setTimeout(function(){ // 同步代码,语句1
  2. console.log('setTimeout') // 宏任务,语句2
  3. });
  4. new Promise(function(resolve, reject){
  5. console.log('pormise body'); // 同步代码,语句3
  6. resolve(); // 同步代码,语句4
  7. }).then(function(){
  8. console.log('promise then') // 微任务,语句5
  9. });
  10. console.log('main'); // 同步代码,语句6

那么他们运行的规则是怎样的呢?

 

原来,宏任务与微任务,都各自有一个调用队列(先进先出)。

 

遇到宏任务,微任务,则将他们推入各自的调用队列。需要注意的是,同步代码也是宏任务。

 

宏任务执行完一个,则去清空微任务队列,微任务清空后再去执行下一个宏任务。

 

这个过程颇像去医院诊室排队看大夫的情景:

如果有一个病人A叫到号以后,又被大夫安排先去做检查,那么做完检查以后病人A回到诊室门口,可以直接等待当前正在看病的病人B与大夫交谈结束后,将检查结果交给大夫,而不用再次排队。

 

我们来把上面的代码一行一行解读一下:

 

首先定义两个队列,宏任务队列:MacroQqueue, 微任务队列: MicroQueue

 

第一步,先按同步代码顺序运行

同步代码,语句1: 添加一个宏任务,将语句2推入MacroQueue。 // MacroQueue: [{task: 语句2}]

同步代码,语句3: 打印promise body

同步代码,语句4: 添加一个微任务,将语句5推入MicroQueue。 // MicroQueue: [{task: 语句5}]

同步代码,语句6: 打印main。 // 同步代码(宏任务)完成

 

第二步,开始清空微任务队列

微任务: 语句5,从MicroQueue跳出,打印promise then。 // 微任务队列全部清空

 

第三步,开始清空宏任务队列

宏任务:语句2,从MacroQqueue跳出,打印setTimeout // 宏任务队列全部清空

 

第四步:开始清空微任务队列

队列为空...

 

一轮循环完成。开始下一轮,如此循环下去。

 

通过上面的讲解,大家应该能对宏任务,微任务的运行机制有了一定的了解了吧。那么都有哪些常见的宏任务与微任务呢?

请看下表:

 

宏任务 浏览器 nodejs
同步代码
I/O
setTimeout
setInterval
setImmediate
requestAnimationFrame

 

微任务 浏览器 nodejs
process.nextTick
MutationObserver
Promise (async/await)

好了,我们来一道复杂一点的题练练手:

  1. console.log('sync statement 1');
  2. Promise.resolve().then(function() {
  3. console.log('micro task 1');
  4. setTimeout(function() {
  5. console.log('macro task 1');
  6. }, 0);
  7. }).then(function() {
  8. console.log('micro task 2');
  9. });
  10. setTimeout(function() {
  11. console.log('macro task 2')
  12. Promise.resolve().then(function(){
  13. console.log('micro task 3');
  14. })
  15. }, 0)
  16. console.log('sync statement 2');
  17. //输出:
  18. // sync statement 1
  19. // sync statement 2
  20. // micro task 1
  21. // micro task 2
  22. // macro task 2
  23. // micro task 3
  24. // macro task 1

标注一下方便大家分析:

  1. console.log('sync statement 1'); // 同步代码,语句1
  2. Promise.resolve().then(function() { // 同步代码,语句2,注册了一个微任务
  3. console.log('micro task 1'); // 微任务,语句3
  4. setTimeout(function() { // 微任务,语句4,同时注册了一个宏任务
  5. console.log('macro task 1'); // 宏任务,语句5
  6. }, 0);
  7. }).then(function() {
  8. console.log('micro task 2'); // 微任务,语句6
  9. });
  10. setTimeout(function() { // 同步代码,语句7
  11. console.log('macro task 2') // 宏任务,语句8
  12. Promise.resolve().then(function(){ // 宏任务,语句9,同时注册了一个微任务
  13. console.log('micro task 3'); // 微任务,语句10
  14. })
  15. }, 0)
  16. console.log('sync statement 2'); // 同步代码,语句11

js中的宏任务与微任务的更多相关文章

  1. javascript中的宏任务和微任务(一)

    一.宏任务和微任务有哪些 宏任务:setTimeout,setInterval,ajax,dom,宏任务是由浏览器提供的 微任务:promise,async/await,微任务是由es6提供的 二.微 ...

  2. javascript中的宏任务和微任务(二)

    js事件轮询执行顺序总结: 1)所有的同步任务都在主线程上执行,行成一个执行栈. 2)除了主线程之外,还存在一个任务列队,只要异步任务有了运行结果,就在任务列队中植入一个时间标记. 3)主线程完成所有 ...

  3. JS异步之宏队列与微队列

    1. 原理图 2. 说明 JS 中用来存储待执行回调函数的队列包含 2 个不同特定的列队 宏列队:用来保存待执行的宏任务(回调),比如:定时器回调.DOM 事件回调.ajax 回调 微列队:用来保存待 ...

  4. JS中EventLoop、宏任务与微任务的个人理解

    为什么要EventLoop? JS 作为浏览器脚本语言,为了避免复杂的同步问题(例如用户操作事件以及操作DOM),这就决定了被设计成单线程语言,而且也将会一直保持是单线程的.而在单线程中若是遇到了耗时 ...

  5. JavaScript同步模式,异步模式及宏任务,微任务队列

    首先JavaScript是单线程的语言,也就是说JS执行环境中,负责执行代码的线程只有一个.一次只能执行一个任务,如果有多个任务的话, 就要排队,然后依次执行,优点就是更安全,更简单.缺点就是遇到耗时 ...

  6. JS中的执行机制(setTimeout、setInterval、promise、宏任务、微任务)

    1.执行机制 JS 是单线程的,处理 JS 任务(程序)只能一个一个顺序执行,所以 JS 中就把任务分为了同步任务和异步任务.同步的进入主线程先执行,异步的进入Event Table并注册函数,当指定 ...

  7. js的事件循环机制:同步与异步任务(setTimeout,setInterval)宏任务,微任务(Promise,process.nextTick)

    javascript是单线程,一切javascript版的"多线程"都是用单线程模拟出来的,通过事件循环(event loop)实现的异步. javascript事件循环 事件循环 ...

  8. js 宏任务和微任务

    .宏任务(macrotask )和微任务(microtask ) macrotask 和 microtask 表示异步任务的两种分类. 在挂起任务时,JS 引擎会将所有任务按照类别分到这两个队列中,首 ...

  9. js 同步 异步 宏任务 微任务 文章分享

    分享一篇 写的很好的 宏任务 微任务  同步异步的文章 文章原地址: https://juejin.im/post/59e85eebf265da430d571f89 这一次,彻底弄懂 JavaScri ...

随机推荐

  1. 读书笔记——《MySQL DBA 工作笔记》

    关于前言 作者在前言中提出的一些观点很具有参考价值, 梳理完整的知识体系 这是每一个技术流都应该追逐的,完整的知识体系能够使我们对知识的掌握更加全面,而不仅仅局限于点 建立技术连接的思维,面对需求,永 ...

  2. mysql 5.6.44 zip 安装教程(基于windows )

    首先我们从官网下载最新版本的mysql 5.6.44版本,网页地址: ,我的电脑是64位的版本,所以我选择的是64位版本,如果是32位,根据实际情况下载所需要的安装包.

  3. OPTIONS 请求引发的分析

    阅读提纲: 为什么会出现 OPTIONS 请求? 什么情况下会出现 OPTIONS 请求? OPTIONS 请求会发送什么内容? 跨域前端访问后端时,所有的 Ajax HTTP 请求都会先发送一个 O ...

  4. StrGame

    如果先手可以控制一轮必胜或者必败,则先手必胜 如果只有必胜的方法,不能保证必败,则最后一轮的先手获得胜利,倒数第二轮的先手会被后手想办法”被胜利“从而在最后一轮成为后手,必败.倒数第三轮先手故意胜利, ...

  5. NOIP模(ka)拟(chang)测试30 考试报告

    应得分:300 实得分:210 毒瘤卡常出题人,卡掉90分! T1 Return 开个副本数组sort一下,unique去重就可以啦.时间复杂度$ O(nlog2(n)) $ T2 One 其实就是约 ...

  6. sshd服务以及基于口令的远程登陆

    ssh用为客户端,主要进行服务器端的连接:sshd用为服务器端 几个常用的命令: systemctl              ##服务控制命令   systemctl start sshd   ## ...

  7. 迁移桌面程序到MS Store(11)——应用SVG图标

    在传统桌面程序中,对图标的使用大多是直接嵌入JPG或者PNG的图片.在祖传的1366x768分辨率下,并没有什么问题.相对于手机硬件的突飞猛进,也侧面反映了PC行业的落寞和桌面程序开发的不思进取.用3 ...

  8. 还看不懂同事的代码?超强的 Stream 流操作姿势还不学习一下

    Java 8 新特性系列文章索引. Jdk14都要出了,还不能使用 Optional优雅的处理空指针? Jdk14 都要出了,Jdk8 的时间处理姿势还不了解一下? 还看不懂同事的代码?Lambda ...

  9. 网站搭建-2-本地网站搭建-安装Linux虚拟机/ 安装IIS Windows

    搭建网站-1-域名申请参见公众号 生物信息系统(swxxxt) 首先,已经拥有了一个可以正常使用的域名. 之前买了两年的阿里的服务器,由于是Windows的,最后不了了之了(因为当时找的代码都是lin ...

  10. C++中对封装的语法支持——重载运算符

    重载运算符 1.对于自定义类型,编译器不知道运算规则,而重载运算符会将两个对象相加转换为函数调用. 2.运算符重载转换的函数调用,函数名字是固定的规则. (1) 如果重载+号运算符,函数名字就是:op ...