Node.js的事件循环是其实现非阻塞异步操作的核心机制,整个循环被划分为多个不同的阶段,每个阶段都有明确的职责。其中poll阶段是处理异步IO操作的核心环节,直接影响着大多数异步任务的执行顺序。

poll阶段的核心作用
poll阶段主要承担两个核心职责:一是检索新的IO事件,执行与IO相关的回调函数;二是根据情况决定事件循环的后续走向,控制定时器和其他阶段回调的执行时机。
当事件循环进入poll阶段时,首先会检查poll队列中是否有待处理的回调。如果有,就会按照先进先出的顺序依次执行这些回调,直到队列清空或者达到系统设定的执行上限。
poll阶段的执行流程
poll阶段的执行逻辑可以分为以下几个步骤:
- 检查poll队列中是否存在待执行的回调,如果有则执行,直到队列为空或者执行数量达到阈值
- 队列清空后,会检查是否有预设的setImmediate回调,如果有则进入check阶段执行
- 如果没有setImmediate回调,就会检查是否有到达时间的定时器,如果有则回到timer阶段执行定时器回调
- 如果以上条件都不满足,事件循环会停留在poll阶段,等待新的IO事件到来
代码示例验证poll阶段行为
我们可以通过一段简单的代码观察poll阶段和其他阶段的执行顺序:
const fs = require('fs');
// 设置一个100ms后执行的定时器
setTimeout(() => {
console.log('timer阶段回调执行');
}, 100);
// 异步读取文件,回调会在poll阶段执行
fs.readFile('./test.txt', (err, data) => {
if (err) {
console.error('文件读取失败');
return;
}
console.log('poll阶段IO回调执行');
// 在IO回调中设置setImmediate
setImmediate(() => {
console.log('check阶段回调执行');
});
});
// 设置一个0ms的定时器,实际会尽快执行
setTimeout(() => {
console.log('timer阶段0ms回调执行');
}, 0);上述代码的执行逻辑如下:首先两个定时器会被插入timer阶段的队列,然后进入poll阶段执行文件读取的IO回调,输出poll阶段IO回调执行后,因为存在setImmediate回调,事件循环会进入check阶段执行对应回调,之后再回到timer阶段执行到期的定时器回调。
poll阶段和其他阶段的关联
poll阶段不是独立运行的,它和事件循环的其他阶段存在紧密的联动关系:
| 关联阶段 | 联动逻辑 |
|---|---|
| timer阶段 | poll阶段停留等待时,若有定时器到期,会回到timer阶段执行回调 |
| check阶段 | poll阶段队列清空后,若存在setImmediate回调,会直接进入check阶段 |
| pending callbacks阶段 | 某些系统操作的回调会在pending callbacks阶段执行,之后进入poll阶段 |
理解poll阶段的工作机制,能帮你更准确地判断Node.js中异步任务的执行顺序,避免因为对事件循环不熟悉导致的逻辑错误。在实际开发中,涉及到文件IO、网络请求等异步操作时,这些回调大多会在poll阶段被处理,掌握这部分知识对优化异步代码性能也有一定帮助。