看代码:
setTimeout(() => {
console.log(1);
Promise.resolve().then(() => {
console.log(4);
});
});
Promise.resolve().then(() => {
console.log(2);
setTimeout(() => {
console.log(3);
});
});
请输出上面的执行结果:...
浏览器的输出结果:2 1 4 3,
node的输出结果为: 2 1 3 4
这样子就有点疑惑了。。。。
2和1的顺序能理解,但是 4和3的顺序,感觉和自己理解的知识发生冲突了,
我的理解是 代码从上往下执行,
将第一个定时器,放到事件回调队列中,
将第一个peomise.then放到微任务中,
然后执行微任务中的代码,先输出2,
然后将第二个定时器塞入事件回调队列,
此时主线程空闲,在事件回调队列中取出第一个定时器的回调执行,打印1,
然后将promise放到微任务中,
此时就出现分歧了,此时的主线程或者执行栈是从微任务中取出promise来执行打印4还是取出定时器的函数执行打印3;
从结果来看,执行栈选择了打印3,
从饿了么团队的一篇文章中看到这样一句话,node会清空当前所处阶段的队列,即执行所有task,再去检查微任务
望解惑,不论是浏览器或者node端,按我的思路道理都不通啊?
浏览器端循环策略:先执行当前循环宏任务,然后执行这个宏任务下的微任务(h5)。
node循环策略:先执行完本次循环的timer队列,然后执行timer下的微任务(libuv)。
以你这个例子来说,宏任务有:当前代码块,setTimeout,微任务有: Promise。
所以,浏览器端的表现是先执行整个代码块这个宏任务,然后执行这个宏任务下的微任务,即本例当中的(Promise.resolve().then),有且仅有这一个。因此,最先输出2,本次循环结束。
第二次循环找到宏任务(第一个setTimeout),打印出1,然后执行这个宏任务下的所有微任务,打印出4,第二次循环结束。
第三次循环,找到第二个setTimeout, 打印出3,第三次循环结束,所以浏览器端的输出结果是2 1 4 3。
而node端的表现开始与浏览器相同,执行整个代码块,完成微任务,输出2。然后找到第二次循环下的两个timer,依次输出 1,3.当前层级的timer都执行完毕之后,执行timer微任务 4, 所以node端的输出结果为 2 1 3 4。
总结:
浏览器: 代码块(宏任务)=> 微任务promise(输出2)=> 第一个timer(输出1)=>微任务promise(输出4)=>第二个timer(输出3)
node: 代码块=>微任务promise(输出2)=>第一个timer(输出1)=>第二个timer(输出3)=>微任务promise(输出4)