Demystifying the JavaScript Event Loop: A Comprehensive Guide
Introduction
The Event Loop is a crucial mechanism in JavaScript (used in both browsers and NodeJS runtime environments). Despite JavaScript being single-threaded, the Event Loop enables it to handle multiple asynchronous tasks (like `setTimeout`, `setInterval`, `fetch`, etc.) efficiently, similar to how multi-threading works in other programming languages.
Related Components
Call Stack
The Call Stack is a vital concept that explains how the JavaScript engine keeps track of function calls within a program. It operates on a Last In, First Out (LIFO) structure, meaning the last function called is the first one to be executed.
When a function is called, it's added to the Call Stack. Once the function completes execution, it's removed from the Call Stack. This process helps JavaScript execute tasks sequentially, in the order functions are called.
Task Queue
The Task Queue (also known as the Callback Queue, Event Queue, or Macrotask Queue) contains JavaScript tasks scheduled for execution, such as callbacks, timeouts/intervals, network communications, and DOM manipulations (like triggered events). The Event Loop processes these tasks in the order they were added to the queue.
Actually, when executing functions like setTimeout/setInterval, the scheduling of the callback function to execute after a certain period is not handled by the Event Loop but by the Web API (if the execution environment is a Web Browser) or libuv (if the execution environment is NodeJS).
For example, when executing setTimeout(callback, 1000) in a browser, the task of waiting for 1000ms and then placing the callback function into the Task Queue is performed by the Web API. The Event Loop's job then is to check if the Call Stack is empty to push the task into it.
Microtask Queue
The Microtask Queue includes functions that run after the current task completes, such as promises and Mutation Observers. You can add a callback directly to the Microtask Queue using the `queueMicrotask` function.
Note:
The Microtask Queue has a higher priority than the Task Queue. This means that both the Call Stack and Microtask Queue must be empty before the Event Loop moves a task from the Task Queue to the Call Stack for execution.
One Microtask can trigger another Microtask, which can lead to a situation where the Microtask Queue is never empty, causing the application to execute indefinitely.
In simple terms, here's how the Event Loop works:
- When a function/code/task is executed, it is added to the Call Stack.
- Synchronous tasks are executed immediately in the Call Stack.
- Asynchronous tasks are categorized:
- Promises and Mutation Observers are added to the Microtask Queue.
- `setTimeout`, `setInterval`, network communications, and DOM manipulations (like triggering events) are added to the Task Queue (MacroTask Queue).
- If the Call Stack is empty, tasks from the Microtask Queue are moved to the Call Stack for execution.
- Only when both the Call Stack and Microtask Queue are empty, tasks from the Task Queue (MacroTask Queue) are moved to the Call Stack for execution.
Explaining Different Use Cases
Next, let's look at various use cases, ranging from synchronous tasks to asynchronous tasks, all related to the Event Loop.
Call Stack
Consider the following example:
Here's a simple code block where you might already know the outcome, and here's the order in which each task executes in the Call Stack to achieve that result:
3. The `f1` function is added to the Call Stack and the `console.log('f1 executed')` statement runs. |
4. The `f1` function finishes executing and is removed from the Call Stack, then the `console.log('f2 completed')` statement runs. After that, the `f2` function is also removed from the Call Stack. |
You can see that the `f2` function is added to the Call Stack before the `f1` function. Due to the Stack's LIFO (Last In, First Out) mechanism, the `f1` function must be executed and removed from the Stack before the `f2` function.
Call Stack and Microtask Queue
Let's look at the following example:
Here's how the execution order works:
This is how the Event Loop handles tasks in the Call Stack and the Microtask Queue:
1. Execute `console.log(1)` |
2. Add `Promise.resolve()` to the Microtask Queue. Since the function has no name, we'll call it `anonymous` |
3. Execute `console.log(4)` |
4. Move the `anonymous` function to the Call Stack and execute `console.log(2)` |
5. Add the callback function from `.then` to the Microtask Queue, also calling it `anonymous` |
6. Move the `anonymous` function to the Call Stack and execute `console.log(3)`. After execution, the `anonymous` function is removed from the Call Stack |
As mentioned in the theory section, `console.log(1)` and `console.log(4)` are synchronous tasks, so they go into the Call Stack and execute first. Promises are asynchronous tasks, so they are executed later. They go into the Microtask Queue, and once the Call Stack is empty, each task in the Microtask Queue is moved to the Call Stack and executed in order.
Call Stack, Microtask Queue and Task Queue (MacroTask Queue)
Here is the execution order result:
And this is how the Event Loop executes tasks in the Call Stack along with the Microtask Queue and Task Queue (MacroTask Queue):
1. Execute `console.log(1)` |
2. Add `setTimeout` function to the Task Queue; since the function has no name, it is called `anonymous` |
3. Add `Promise.resolve()` to the Microtask Queue; since the function has no name, it is called `anonymous` |
6. Add the callback in `.then` to the Microtask Queue; since the function has no name, it is called `anonymous` |
7. Move the `anonymous` function from the Microtask Queue to the Call Stack and execute `console.log(4)` |
8. Move the `anonymous` function from the Task Queue to the Call Stack and execute `console.log(2)`; after executing, the `anonymous` function is also removed from the Call Stack |
Based on the theory of components related to the Event Loop to explain the execution order:
- First, synchronous tasks are executed, including `console.log(1)` and `console.log(5)`
- Next, the task to be executed in setTimeout is placed in the Task Queue (MacroTask Queue)
- The task to be executed in the Promise is placed in the Microtask Queue
- The Microtask Queue has a higher priority than the Task Queue (MacroTask Queue), so tasks in the Microtask Queue are moved to the Call Stack to execute first. Once both the Call Stack and Microtask Queue are empty, the task from the Task Queue (MacroTask Queue) is moved to the Call Stack to continue execution.
Application of the Event Loop to explain issues
You can see that the Event Loop is used to handle all code execution in JavaScript, from synchronous to asynchronous tasks. Understanding the Event Loop will help you understand the execution order of the working flow in your project, allowing you to detect and resolve issues related to the application of too many asynchronous tasks effectively.
Next, I will provide 2 examples of JavaScript coding that frequently appear in interviews as follows.
Example 1: Explain and provide the result of the following block code
This is a common question used to test your understanding of the Event Loop theory. To know the execution result of your program, you just need to understand the execution order of tasks in the Call Stack, Microtask Queue, and Task Queue (Macrotask Queue) to analyze each line of code for the result. The task execution process is as follows:
- `console.log(1)` is a synchronous task that will execute first.
- `setTimeout` is placed in the Task Queue.
- `new Promise` differs from `Promise.resolve` because it will execute immediately upon initialization, so it will continue to execute the line `console.log(4)`. After that, `setTimeout` is placed in the Task Queue. Since the `resolve` function has not been called, the subsequent `.then` will not execute yet.
- `console.log(8)` is executed next.
- At this point, there are only 2 tasks in `setTimeout`, so each task is taken and placed in the Call Stack to execute. The line `console.log(2)` is executed first, followed by placing `Promise.resolve()` in the Microtask Queue.
- Now there is 1 task in the Task Queue and 1 task in the Microtask Queue, and since the Microtask Queue has a higher priority than the Task Queue, it is executed first, so the line `console.log(3)` is executed.
- There is only 1 `setTimeout` left in the Task Queue, so `console.log(5)` is executed.
- After executing `resolve(6)`, it will be placed in the Microtask Queue.
- Grab the task from the Microtask Queue to execute the final `.then` and execute `console.log(7)`.
- There is still 1 `setTimeout` which is placed in the Task Queue, then it is taken into the Call Stack to execute the line `console.log(res)` (where `res` is 6).
The result is as follows:
Example 2: Fix the error so that the following program does not block.es not block."
This block of code is implemented using React to calculate the sum from 1 to 9e9 (9*10^9). You may realize that when calling the `start` function, executing in the `for` loop will take a lot of processing time, causing the main thread to be blocked, making the application unable to continue functioning.
Once the cause is identified as the `for` loop having to execute 9e9 times, the solution to address this is to break down the number of loop iterations (for example, looping only 1e7 (1*10^7) times each time). After each small loop, update the values of the `total` and `time` variables, then continue executing small loops until finished. Thus, the code after the fix will look like this.
You can see that the code is divided so that each iteration runs only 1e7 times, then it will call the `setTimeout` function for the next iteration if it has not yet completed 9e9 times. When using the `setTimeout` function, as mentioned theoretically, it will be placed in the Task Queue (Macrotask Queue), and the execution order will be Call Stack -> Microtask Queue -> Re-render (on browser) -> Task Queue (Macrotask Queue). Thus, after every 1e7 iterations, the next iteration is placed in the Task Queue, so it will not get blocked; setting the values for total and time will also be shown on the UI as usual.
Bonus question
After reading this article, if you think you understand Event Loop, try to analyze to see what the result of the following block of code is. This type of question often appears in JavaScript interviews to test your understanding of Event Loop. If you can answer it, you are surely ready to tackle issues related to Event Loop.
Happy coding!
Comments
Post a Comment