*微任务: promise.then < process.nextTick(先) 1. 主执行栈队列 2. timer队列: setTimeout/setInterval // 到时间后,将任务加入timer队列;没有到时间,且check队列为空,就切换到poll队列等待 3. poll队列: i/o接口,fs.readFile //如果check队列为空,会在此阶段等待定时器到达 4. check队列: setImmediate
1. 执行顺序说明
node10及之前和node11之后的“微任务队列清空条件”不同: 1)node10及之前的版本,队列切换时才会清空微任务队列 2)node11及之后的版本,每执行一个宏任务就清空微任务队列(同浏览器)
1. node V10(每次切换都清空队列)
1. 清空主执行栈队列
2. 清空微任务队列。
3. 按照timer->poll->check队列执行,只要执行队列切换就清空微任务队列。则主执行栈切换到timer队列,也会先清空微任务队列。
4. 定时器时间到达,清空所有的timer队列。
5. 如果定时器时间未到达,到poll队列,查看check队列是否有任务,如果有,清空。否则,在poll队列等待。
6. 轮训timer队列是否有任务。
2. node V11+ (执行一次宏任务就清空微任务队列)
同浏览器事件环
1. 清空主执行栈队列
2.清空微任务队列
3.按照timer->poll-check队列。如果timer定时器时间不到,进入poll队列执行,然后检查check队列,有任务的话,先执行第一个,然后检查微任务队列,如果有任务则清空,再继续执行check队列的其他任务,每执行一次都要清空微任务队列;否则继续等待,直到定时器到达。
4.如果定时器时间到达,如果timer队列有多个任务,先执行第一个,然后取清空微任务队列,然后继续执行,每执行一个timer中的任务就清空一次微任务队列。
5.再进入poll队列,依timer->poll-check轮训
2. 应用
示例1:
setTimeout(() => { console.log('timeout') }) setImmediate(() => { console.log('immediate') }) // node命令执行后,先后顺序不一定。根据setTimeout定时器的到达时间快慢。 // 如果setTimout回调函数先进入队列,先执行;否则setImmediate先执行
示例2:
const fs = require('fs'); fs.readFile('1.txt', 'utf-8', function() { process.nextTick(() => { console.log('nexttick') }) setTimeout(() => { console.log('settimeout') }) setImmediate(() => { console.log('setImmediate') }) }) // 运行结果如下: nexttick setImmediate //按照事件环,一定先执行;因为fs是poll队列,poll队列->check队列 settimeout
示例3:
process.nextTick(() => { console.log('nexttick') }) setTimeout(() => { console.log('settimeout1') }) setTimeout(() => { console.log('settimeout2') }) setImmediate(() => { console.log('setImmediate') process.nextTick(() => { console.log('immediate->nexttick') }) }) const fs = require('fs'); fs.readFile('1.txt', 'utf-8', function() { process.nextTick(() => { console.log('fs->nexttick') }) setTimeout(() => { console.log('fs->settimeout') }) setImmediate(() => { console.log('fs->setImmediate') }) }) fs.readFile('1.txt', 'utf-8', function() { process.nextTick(() => { console.log('fs2->nexttick') }) setTimeout(() => { console.log('fs2->settimeout') }) setImmediate(() => { console.log('fs2->setImmediate') }) }) // node V10运行结果如下: nexttick settimeout1 settimeout2 fs->nexttick fs2->nexttick setimmediate fs->setImmediate fs2->setImmediate immediate->nexttick fs->settimeout fs2->settimeout
// node V11运行结果如下
nexttick
settimeout1
settimeout2
fs->nexttick
fs2->nexttick
setimmediate
immediate->nexttick
fs->setImmediate
fs2->setImmediate
fs->settimeout
fs2->settimeout