看代码:setTimeout(()=>{console.log(1);Promise.resolve().then(()=>{console.log(4);});});Promise.resolve().then(()=>{console.log(2);setTimeout(()=>{console.log(3);});});请输出上面的执行结果:...浏览器的输出结果:2143,node的输出结果为:2134这样子就有点疑惑了。。。。2和1的顺序能理解,但是4和3的顺序,感觉和自己理解的知识发生冲突了,我的理解是代码从上往下执行,将第一个定时器,放到事件回调队列中,将第一个peomise.then放到微任务中,然后执行微任务中的代码,先输出2,然后将第二个定时器塞入事件回调队列,此时主线程空闲,在事件回调队列中取出第一个定时器的回调执行,打印1,然后将promise放到微任务中,此时就出现分歧了,此时的主线程或者执行栈是从微任务中取出promise来执行打印4还是取出定时器的函数执行打印3;从结果来看,执行栈选择了打印3,从饿了么团队的一篇文章中看到这样一句话,node会清空当前所处阶段的队列,即执行所有task,再去检查微任务望解惑,不论是浏览器或者node端,按我的思路道理都不通啊?
2 回答
梵蒂冈之花
TA贡献1900条经验 获得超5个赞
浏览器端循环策略:先执行当前循环宏任务,然后执行这个宏任务下的微任务(h5)。node循环策略:先执行完本次循环的timer队列,然后执行timer下的微任务(libuv)。以你这个例子来说,宏任务有:当前代码块,setTimeout,微任务有:Promise。所以,浏览器端的表现是先执行整个代码块这个宏任务,然后执行这个宏任务下的微任务,即本例当中的(Promise.resolve().then),有且仅有这一个。因此,最先输出2,本次循环结束。第二次循环找到宏任务(第一个setTimeout),打印出1,然后执行这个宏任务下的所有微任务,打印出4,第二次循环结束。第三次循环,找到第二个setTimeout,打印出3,第三次循环结束,所以浏览器端的输出结果是2143。而node端的表现开始与浏览器相同,执行整个代码块,完成微任务,输出2。然后找到第二次循环下的两个timer,依次输出1,3.当前层级的timer都执行完毕之后,执行timer微任务4,所以node端的输出结果为2134。总结:浏览器:代码块(宏任务)=>微任务promise(输出2)=>第一个timer(输出1)=>微任务promise(输出4)=>第二个timer(输出3)node:代码块=>微任务promise(输出2)=>第一个timer(输出1)=>第二个timer(输出3)=>微任务promise(输出4)
红糖糍粑
TA贡献1815条经验 获得超6个赞
setImmediate(()=>{console.log(1);process.nextTick(()=>{console.log(4);});});process.nextTick(()=>{console.log(2);setImmediate(()=>{console.log(3);});});这段代码是那遍文章里面提到的,文章说是输出2134但是你会发现把它扔到node的最新版中输出是2143旧一点的版本是2134这个表明node的异步执行机制已经改了,不是文章所说那样了
添加回答
举报
0/150
提交
取消