6个主要阶段
当js同步脚本运行完后,如果有异步操作还没有完成,node就将进入事件循环,像http.createServer.listen,fs.readFileAsync等操作都会使node进入事件循环,没有的话node将直接退出。
timers pending callbacks idle, prepare poll check close callbacks
┌───────────────────────┐
┌ │ timers │
│ └──────────┬────────────┘
│ ┌──────────┴────────────┐
│ │ pending callbacks │
│ └──────────┬────────────┘
│ ┌──────────┴────────────┐
│ │ idle, prepare │
│ └──────────┬────────────┘ ┌───────────────┐
│ ┌──────────┴────────────┐ │ incoming: │
│ │ poll │ ─────┤ connections │
│ └──────────┬────────────┘ │ data, etc. │
│ ┌──────────┴────────────┐ └───────────────┘
│ │ check │
│ └──────────┬────────────┘
│ ┌──────────┴────────────┐
└──┤ close callbacks │
└───────────────────────┘
每个阶段都维护一个任务队列,进行下一阶段的最低条件是清空本阶段任务队列的任
这个阶段执行 setTimeout(callback) 和 setInterval(callback) 预定的 callback; 注意:setTimeout(callback,0) 在node中会被强制转换为setTimeout(callback,1)
系统底层操作的回调,如TCP错误
闲置预备阶段,仅供内部使用,不做讨论
处理I/O事件的回调,适当时候,Node将在这里堵塞
setImmediate的回调在此执行
关闭的回调,close事件的监听回调
主要任务 1.计算定时器应该阻塞的时间 2.执行该阶段队列的所有回调函数(I/O回调) 3.如果队列为空,也就是没有工作可做,这是如果有setImmediate设置的回调存在,就会直接结束该阶段,不会等待响应的I/O回调,如果没有setImmediate设置的回调队列,就会等待I/O操作 举个例子 比如有一个设置了100ms后执行的定时器,同时有一个I/O操作,交给内核,内核正在读取文件,当事件循环开始时,timers阶段显示时间没到100ms,跳过进入poll阶段,此时文件还没有读完,但因为poll阶段检测到也没到定时器的100ms,所以即使进行下一轮的事件循环还是会跳过timers阶段,所以决定等待文件读取的操作,就是堵塞在poll阶段,直到100ms,此时如果文件读取完成,就会把文件读取的回调执行完,再进入下一轮事件循环,执行timers的定时器
但有一个情况会有所不同,就是在cheak阶段的队列不为空,即有setImmediate设置的回调,此时的poll阶段就不会等待I/O操作,而是会直接清空该阶段的任务队列再进入cheak阶段,清空cheak阶段的任务队列。
用于不同,setImmediate意为在本轮I/O操作后马上执行回调,setTimeout则是一段时间后,尽可能快的执行回调
独立维护一个队列,在每个阶段结束后,都会优先清空该队列
setTimeout,setInterval,setImmediate I/O回调 同步脚本
process.nxetTick promise.then catch finall 注意:process.nextTick优先级大于promise
每处理一个宏任务都会处理掉所有微任务
原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。
如有侵权,请联系 cloudcommunity@tencent.com 删除。
原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。
如有侵权,请联系 cloudcommunity@tencent.com 删除。