关于前端:宏任务和微任务的理解

一、什么是宏工作和微工作?

首先,咱们要先理解下 Js 。js 是一种单线程语言,简略的说就是:只有一条通道,那么在工作多的状况下,就会呈现拥挤的状况,这种状况下就产生了 ‘多线程’ ,然而这种“多线程”是通过单线程模拟的,也就是假的。那么就产生了同步工作和异步工作。

二、JS为什么要辨别宏工作和微工作?

(1)js是单线程的,然而分同步异步\
(2)微工作和宏工作皆为异步工作,它们都属于一个队列\
(3)宏工作个别是:script、setTimeout、setInterval、postMessage、MessageChannel、setImmediate(Node.js 环境)\
(4)微工作:Promise.then、Object.observe、MutationObserver、process.nextTick(Node.js 环境)\
(5)先执行同步再执行异步,异步遇到微工作,先执行微工作,执行完后如果没有微工作,就执行下一个宏工作,如果有微工作,就按程序一个一个执行微工作

三、宏工作、微工作有哪些?

  • 宏工作个别是:script、setTimeout、setInterval、postMessage、MessageChannel、setImmediate(Node.js 环境)
  • 微工作:Promise.then、Object.observe、MutationObserver、process.nextTick(Node.js 环境)

四、宏工作、微工作是怎么执行的?

执行程序:先执行同步代码,遇到异步宏工作则将异步宏工作放入宏工作队列中,遇到异步微工作则将异步微工作放入微工作队列中,当所有同步代码执行结束后,再将异步微工作从队列中调入主线程执行,微工作执行结束后再将异步宏工作从队列中调入主线程执行,始终循环直至所有工作执行结束。

这里容易产生一个谬误的意识:就是微工作先于宏工作执行。实际上是先执行同步工作而后在执行异步工作,异步工作是分宏工作和微工作两种的。

案例

例一:

setTimeout(function(){
    console.log(1);
});
new Promise(function(resolve){            
    console.log(2);
    resolve();
}).then(function(){            
    console.log(3);
}).then(function(){
    console.log(4)
});         
console.log(5);
// 2 5 3 4 1

剖析

  1. 遇到setTimout,异步宏工作,放入宏工作队列中
  2. 遇到new Promise,new Promise在实例化的过程中所执行的代码都是同步进行的,所以输入2
  3. Promise.then,异步微工作,将其放入微工作队列中
  4. 遇到同步工作console.log(5);输入5;主线程中同步工作执行完
  5. 从微工作队列中取出工作到主线程中,输入3、 4,微工作队列为空
  6. 从宏工作队列中取出工作到主线程中,输入1,宏工作队列为空

例二:

setTimeout(()=>{
    new Promise(resolve =>{
        resolve();
    }).then(()=>{
        console.log('test');
    });
    console.log(4);
});
new Promise(resolve => {
    resolve();
    console.log(1)
}).then( () => {
    console.log(3);
    Promise.resolve().then(() => {
        console.log('before timeout');
    }).then(() => {
        Promise.resolve().then(() => {
          console.log('also before timeout')
        })
    })
})
console.log(2);
//1,2,3,before timeout,also before timeout,4,test;

剖析

  1. 遇到setTimeout,异步宏工作,将() => {console.log(4)}放入宏工作队列中;
  2. 遇到new Promise,new Promise在实例化的过程中所执行的代码都是同步进行的,所以输入1;
  3. 而Promise.then,异步微工作,将其放入微工作队列中
  4. 遇到同步工作console.log(2),输入2;主线程中同步工作执行完
  5. 从微工作队列中取出工作到主线程中,输入3,此微工作中又有微工作,Promise.resolve().then(微工作a).then(微工作b),将其顺次放入微工作队列中;
  6. 从微工作队列中取出工作a到主线程中,输入 before timeout;
  7. 从微工作队列中取出工作b到主线程中,工作b又注册了一个微工作c,放入微工作队列中;
  8. 从微工作队列中取出工作c到主线程中,输入 also before timeout;微工作队列为空
  9. 从宏工作队列中取出工作到主线程,此工作中注册了一个微工作d,将其放入微工作队列中,接下来遇到输入4,宏工作队列为空
  10. 从微工作队列中取出工作d到主线程 ,输入test,微工作队列为空

例三:

console.log(1)
setTimeout(function() {
    console.log(2)
}, 0)
const p = new Promise((resolve, reject) => {
    resolve(4)
})
p.then(data => {
    console.log(data)
})
console.log(3)
//1,3,4,2

剖析

  1. 遇到同步工作console.log(1);输入1;
  2. 遇到setTimeout 异步宏工作,放入宏工作队列中;
  3. 遇到 Promise,new Promise在实例化的过程中所执行的代码都是同步进行的,但因为new Promise没有输入事件,所以接着执行遇到.then;
  4. 执行.then,异步微工作,被散发到微工作Event Queue中;
  5. 遇到同步工作console.log(3);输入3;
  6. 主线程中同步工作执行完,从微工作队列中取出工作到主线程中,p.then 输入4,微工作执行结束,工作队列为空;
  7. 开始执行宏工作setTimeout 输入2,宏工作队列为空;

例四:

console.log(1)
setTimeout(function() {
    console.log(2)
    new Promise(function(resolve) {
        console.log(3)
        resolve()
    }).then(function() {
        console.log(4)
    })
})

new Promise(function(resolve) {
    console.log(5)
    resolve()
}).then(function() {
    console.log(6)
})

setTimeout(function() {
    console.log(7)
    new Promise(function(resolve) {
        console.log(8)
        resolve()
    }).then(function() {
        console.log(9)
    })
})
console.log(10)
//1,5,10,6,2,3,4,7,8,9

剖析:

  1. 遇到同步工作console.log(1);输入1;
  2. 遇到setTimeout 异步宏工作,放入宏工作队列中;
  3. 遇到 Promise,new Promise在实例化的过程中所执行的代码都是同步进行的,所以输入5,所以接着执行遇到.then;
  4. 执行.then,异步微工作,被散发到微工作Event Queue中;
  5. 遇到setTimeout,异步宏工作;放入宏工作队列中;
  6. 遇到同步工作console.log(10);输入10,主线程中同步工作全副执行完;
  7. 从微工作队列中取出工作到主线程中,输入6;
  8. 在从宏工作队列中取出工作到主线程中,执行第一个setTimeout,输入2,3,4(在宏工作中执行同步,同步,异步微工作);
  9. 在执行第二个setTimeout,输入7,8,9(和8同理);

例五:

new Promise((resolve, reject) => {
    resolve(1)
    new Promise((resolve, reject) => {
        resolve(2)
    }).then(data => {
        console.log(data)
    })
}).then(data => {
    console.log(data)
})
console.log(3)
//3,2,1

剖析:

  1. 遇到Promise,new Promise在实例化的过程中所执行的代码都是同步进行的,但因为new Promise没有输入事件,所以接着往下执行遇到new Promise没有输入事件再接着往下执行遇到.then,异步微工作,被散发到微工作Event Queue中,再接着 .then 放入微工作
  2. 遇到同步工作console.log(3) 输入3,主线程中同步工作执行完;
  3. 从微工作队列中取出工作到主线程中,输入2,1,微工作执行结束,工作队列为空。

总结

微工作和宏工作的执行程序是先执行同步工作,先执行同步后异步,异步分为宏工作和微工作两种,异步遇到微工作先执行微工作,执行完后如果没有微工作,就执行下一个宏工作。

该文章首发于掘金平台

最初感激大家浏览⭐️⭐️⭐️,如果喜爱对你也有帮忙,欢送点赞和留言哟💕💕💕

评论

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

这个站点使用 Akismet 来减少垃圾评论。了解你的评论数据如何被处理