Event Loop Phases in Node.js
The event loop has several different phases to it and each one of these phases maintains a queue of callbacks that are to be executed.
Callbacks are destined for different phases based on how they are used by the application.
Phase 1: Poll
- Starting point of Node.js application
- Most of the application code executes in this phase.
- The poll phase executes I/O-related callbacks.
Phase 2: Check
- In this phase, callbacks that are triggered via
setImmediate()
are executed.
Phase 3: Close
- This phase executes callbacks triggered via
EventEmitter close events
. - For example, when a net.Server TCP server closes, it emits a close events that runs in this phase.
Phase 4: Timers
- In this phase, callbacks triggered via
setTimeout()
andsetInterval()
are executed.
Phase 5: Pending
- Special system events are run in this phase, like when a net.Socket TCP soccer throws an
ECONNREFUSED
error.
Microtask queues
Apart from these, there are two special microtask queues that can have callbacks added to them while a phase is running.
- The first microtask queue handles callbacks registered using
process.nextTick()
. - The second microtask queues handles
promises
that reject or resolve.
Execution Priority and order
- Callback in the microtask queues take priority over callbacks in the phase's normal queue.
- Callbacks in the next tick microtask queue run before callbacks in the promise microtask queue.
- When the application starts running, the event loop is also started and the phases are handled one at a time. Node.js adds callbacks to different queues as appropriate while the application runs.
- When the event loop gets to a phase, it will run all the callbacks in the phase's queue. Once all the callbacks in a given phase are executed, the event loop then moves on to the next phase.
Let's see one code example:
const fs = require('fs');
setImmediate(() => console.log(1));
Promise.resolve().then(() => console.log(2));
process.nextTick(() => console.log(3));
fs.readFile(__filename, () => {
console.log(4);
setTimeout(() => console.log(5));
setImmediate(() => console.log(6));
process.nextTick(() => console.log(7));
});
console.log(8);
Output will be 8 3 2 1 4 5 6 5
Let's see what is happening behind the scene:
Code execution starts off executing line by line in the poll phase.
Step 1: The fs module is required.
Step 2: The setImmediate()
call is run and its callback is added to the check queue.
Step 3: the promise
resolves, adding callback to the promise microtask queue
.
Step 4: process.nextTick()
runs next, adding its callback to the next tick microtask queue
.
Step 5: The fs.readFile()
tells Node.js to start reading the file, placing its callback in the poll queue
once it is ready.
Step 6: Finally console.log(8)
is called and 8 is printed to the screen.
That's it for the current stack.
- Now, the two microtask queues are consulted. The next tick microtask queue is always checked first, and callback with output 3 is called. Since, there is only one callback in the next tick microtask queue, the promise microtask queue is checked next and callback with output 2 is executed. That finished the two micro-task queues and the current poll phase is completed.
- Now, the event loop enters the check phase. This phase has callback 1 in it, which is then executed. Both the microtask queues are empty at this point, so the check phase ends.
- The close phase is checked next but is empty, so the loop continues. The same happens with the timers phase and the pending phase, and the event loop continues back around to the poll phase.
Once it is back in the poll phase, the application doesn't have much else going on, so it basically waits until the file has finished being read. Once that happens, the fs.readFile()
callback is run.
- The number 4 is immediately printed since it's the first line in the callback.
- next, the
setTimeout()
call is made and callback 5 is added to the timers queue. - The
setImmediate()
call happens next, adding callback 6 to the check queue. - Finally, the
process.nextTick()
call is made, adding callback 7 to the next ticket microtask queue.
The poll phase is now finished and the microtask queues are again consulted.
- Callback 7 runs from the next tick queue,
- The promise queue is consulted and found empty, and the poll phase ends.
- Again the event loop enters to the check phase where callback 6 is encountered. The number is printed and microtask queues are determined to be empty and the phase ends.
- The close phase is checked again and found empty.
- Finally, the timers phase is consulted and callback 5 is executed and prints 5 on the console.
- Once that's done, the applications doesn't have any more work to do and it exits.
As we know, Node.js runtime environment is single-threaded. Running too much code in a single stack will stall the event loop and prevent other callbacks from firing.
To prevent this event loop starving situation, you can break your CPU-heavy operations up across multiple stacks.
For example, if you are processing 1000 data records, you can consider breaking down into 10 batches of 100 records, using setImmediate()
at the end of each batch to continue processing the next batch.
Another option is forking a new child process and offload processing to it. But never break up such work using process.nextTick()
. Doing so will lead to a microtask queue that never empties and your application will be trapped in the same phase forever. The runtime won't throw any error instead it will remain a zombie process that eats through CPU.
Reference: Distributed Systems with Node.js (Book)