Understanding the Node.js Event Loop

Mar 8, 2025

Ever heard someone say, "Node.js is single-threaded but can handle thousands of concurrent requests" and wondered, HOW? 🀯

The answer lies in the Event Loopβ€”the heart of Node.js' asynchronous behavior. Today, we’re breaking it down in a simple, no-BS way.


πŸš€ What is the Event Loop?

The Event Loop is what makes asynchronous, non-blocking I/O possible in JavaScript.

Imagine you're at a restaurant:

  • The waiter (Event Loop) takes multiple orders (incoming requests).
  • Instead of waiting for one dish to cook (blocking), they move on to the next customer.
  • When a dish is ready, they serve it (callback execution).

Node.js works the same wayβ€”handling multiple tasks without getting stuck.


πŸ”₯ The 6 Phases of the Event Loop

The Event Loop is made up of six phases, running in cycles:

1️⃣ Timers β†’ Runs setTimeout() and setInterval() callbacks.
2️⃣ Pending Callbacks β†’ Executes deferred I/O callbacks.
3️⃣ Idle, Prepare β†’ Used internally by Node.js (not important for devs).
4️⃣ Poll β†’ Processes new I/O events (e.g., reading files, network requests).
5️⃣ Check β†’ Executes setImmediate() callbacks.
6️⃣ Close Callbacks β†’ Runs cleanup operations (e.g., socket.on('close')).

Each cycle runs all pending tasks before moving on to the next phase.


πŸ›  Hands-on Example: Timers vs. Immediate

Let’s compare setTimeout() and setImmediate():

setTimeout(() => console.log("⏳ Timeout callback"), 0);
setImmediate(() => console.log("πŸš€ Immediate callback"));
console.log("🎬 Script start");

πŸ§ͺ What happens?

🎬 Script start
πŸš€ Immediate callback
⏳ Timeout callback

πŸ” Why?

  • setImmediate() runs after the poll phase, making it execute sooner.
  • setTimeout(..., 0) waits until the next timer phase.

⚑ Async I/O in the Event Loop

I/O operations like reading a file use the Poll Phase.

import { readFile } from "fs";

console.log("🎬 Start reading file...");

readFile("example.txt", "utf8", (err, data) => {
  console.log("πŸ“– File read completed!");
});

console.log("πŸš€ Other tasks running...");

πŸ§ͺ What happens?

🎬 Start reading file...
πŸš€ Other tasks running...
πŸ“– File read completed!

πŸ” Why?

  • readFile() is asynchronous, so Node.js doesn’t block execution.
  • The Event Loop continues with other tasks until the file is ready.

🎭 Microtasks: process.nextTick() vs. Promises

Node.js has a Microtask Queue, which runs before the next Event Loop phase.

setTimeout(() => console.log("⏳ setTimeout"), 0);
setImmediate(() => console.log("πŸš€ setImmediate"));
process.nextTick(() => console.log("πŸ‡ nextTick"));
Promise.resolve().then(() => console.log("✨ Promise"));
console.log("🎬 Script start");

πŸ§ͺ What happens?

🎬 Script start
πŸ‡ nextTick
✨ Promise
⏳ setTimeout
πŸš€ setImmediate

πŸ” Why?

  • process.nextTick() and Promises run before any timers or immediate tasks.
  • setTimeout() and setImmediate() wait for their respective phases.

🎯 Why Does This Matter?

Understanding the Event Loop helps you:

  1. Write non-blocking code πŸš€ β†’ Avoid blocking the main thread with CPU-heavy tasks.
  2. Optimize performance πŸ”₯ β†’ Know when to use setImmediate(), setTimeout(), and process.nextTick().
  3. Debug async issues πŸ•΅οΈβ€β™‚οΈ β†’ Fix unexpected delays in execution.
  4. Handle large-scale applications πŸ“‘ β†’ Scale up your Node.js backend efficiently.
  5. Avoid callback hell πŸ˜΅β€πŸ’« β†’ Know when to use async/await vs. callbacks.

TL;DR: The Event Loop is why Node.js can handle thousands of requests without breaking a sweat. Mastering it makes you a better developer! πŸš€


πŸ”— Final Thoughts

  • The Event Loop is what makes Node.js asynchronous.
  • Timers (setTimeout) and Immediate (setImmediate) behave differently.
  • I/O tasks go through the Poll Phase.
  • Microtasks (nextTick & Promises) run before timers.

Understanding this makes you a better Node.js developer! πŸš€

Let me know if you have any questions! 😊πŸ”₯

diogoaoliveira.com