Hey there 👋,
Before diving into the world of asynchronous events, lets first understand what happens when you execute a function? where do they get stored in memory? How does the interpreter knows the order of the functions to be executed? Can we run multiple functions at a time?
Without making any further delay, lets get started…
The call stack
The mechanism JS interpreter uses to keep track of its place in a script that calls multiple functions.
Ever wondered how JS “knows” what function is currently being executed and what functions need to be called next? with the help of call stack.
How does it work?
- When a script calls a function, interpreter adds it to the call stack and then starts carrying out the function.
- Any functions that are called by that function are added to the call stack further up, and run where their calls are reached.
- When the current function is finished, interpreter takes it off the stack and resumes execution where it left off in the last code listing.
JS is single threaded
At any given point in time, a single JS thread is running atmost one line of JS code. Its not multi tasking, it does one thing at a time.
User may not be able to do another task until one is completed.
|
|
You wont get output of 2nd console.log until alert hi is closed.When we try to access data from a database or an api or trying to save something or to set timer, we cant let user to keep on waiting until we get the data. Don’t we have any workaround for this? We do Asynchronous callbacks.
How Asynchronous callbacks work?
Pass a callback function for the processes that take time and those functions will be executed at the appropriate time.
Set timer for 3 seconds.
|
|
But how does JS know to set a timer for 3 seconds if it can only do 1 thing at a time? how does it remember to wait for 3 seconds and call that function after 3 sec?
The trick here is The browser does the work. JS is not same as our browser, it is a language that is implemented in our browser. Generally browsers like chrome, safari and IE are written in a different programming language (C++ here for those 3).
Browser itself is capable of doing certain tasks, where JS sucks at or things that take time are handed off to the browser.
Note: JS is not keeping track of timer , it is not sending an API request. The browser handles that
But how does browser do that?
Browsers come with web API that are able to handle certain tasks in the background(like making requests or setTimeOut). The JS call stack recognizes these web API functions and passes them off to the browser to take care. Once the browser finishes those tasks they return and are pushed onto the stack as callback.
|
|
Once the browser finishes of setTimeOut it asks JS to run that callback . It places that callback on the call stack so that JS knows to run that function. JS doesnt keep the time. Same thing with making request, JS doesnt make the request itself the browser and the web api do.
Concurrency and the event loop
The reason why we can do things concurrently is that browser is more than just runtime.
This diagram basically looks identical to Node’s architecture, instead of Web apis we have c++ apis(because browser is implemented in c++) and the threading being hidden from us is done by c++.
The setTimeOut api is provided by browser. It doesnt live in V8 source, its the extra we are getting in the environment we are running the javascript
Now that setTimeOut is out of call stack we can execute other lines of code.
The timer in web api cant just start modifying our code, it cant chuck stuff on to the stack when its ready if it did, it would randomly appear in middle of the queue. So any of the webapis pushes the callbacks onto task queue
when its done.
Event loop
Event loop has the one simple job. Its job is to look at the stack and look at the task queue. If the stack is empty,it takes the first thing on the queue and pushes it onto the stack which effectively runs it.
Note:
Event loop pushes the callback onto stack, only when its empty.
- The above used images above are taken from this video
Note:
AJAX requests, DOM, Event Listeners, setTimeOut etc are handled by Web APIs. Here are the list of web apis
Also check out video on event loops for more info.
To visualize how things work in the background check this tool
Now we know why callbacks are so important in javascript. We pass a callback in to certain functions that the browser takes over.
But callbacks are not perfect, they can get messy very quickly.
In the below example, you dont need to worry what transform
and translate
are. Just imagine there is a button on the page, every time the below function executes, it moves right by 100px
|
|
|
|
Now lets say i want to move this button 5 levels deeper for every 1 sec. We need to nest this function .(move 5 times towards right)
|
|
Let’s refactor this
|
|
Even this doesnt look much scarier. Lets throw another callback if we move off the screen. previously we had only one branch going down(moveX-> moveX -> moveX …).
Now we have two call backs for one function. Often this pattern is used we send a http request, if it is success then run one callback, if its a failure run another.
it would look like this..
|
|
The callback hell
|
|
This is where promises come in. Promises allow us to rewrite a function like this without doing all of this crazy nesting. It makes the code more easier to read
If you want to know more about promises, refer Promises Explained.
Hope that was not convoluted and crazy. Until next time, Happy learning 🎉 💻
If you found this helpful, please give a shoutout to @gsavitha_ and share this article to help others. For more articles like this, subscribe to my Newsletter and get the latest updates straight to your inbox.