This post will briefly discuss the mechanism of JavaScript runtime, and three ways to implement asynchronous programming in JavaScript.
Event Loop
Although it is not strictly correct that JavaScript is a single-threaded laungage because either in Browser or Nodejs there might be multiple threads running, JavaScipt script is truly executed in a single thread in runtime, which means it can only do one thing at a time. In intuition then, how it can be asynchronous? The key is Event Loop, a never-bloking and endless loop which keeps waiting -> checking -> executing -> waiting
tasks from a message queue. It looks like this:
1 | while (queue.waitForMessage()) { |
In JavaSsript, whenever a function runs, it is not executed immediately until all other previous functions are completed. JavaScript uses a message queue to store tasks to be processed. In Browser, for example, when you click a button, the event listener just put the function triggered by the click to the message queue. Only after all functions pre-enqueued are processed, this functions will be executed. So the answer to the counter-intuition is: JavaScript implements asynchroncity by manipulating the order of codes processing.
Begin with a simple example:
1 | console.log(1); |
Output:
1 | 1 |
Above is a very simple asynchronous example. setTimeout()
is the key here. It puts console.log(2)
into message queue, whereas console.log(3)
is put into stack and excuted immediately, even if I set second argument as 0
. After stack is empty, event loop checks tasks in queue, and finds console.log(2)
, and then puts it to stack to be processed.
A practice given JavaScript runtime modle is that do not try to make one function takes too long.
After ES6, there are three ways to implement asynchronous programming in JavaScript: Callback, Promise, and Async/await
Callback
A callback is a function passed into another function as an argument by which it is then invoked inside the outer function to complete some kind of routine or action. A callback fuction can be called in either sync or async ways. Here is an example to tell the difference:
1 | function barSync(val, callback) { |
Output:
1 | 1 |
However, in asynchronous context, the callback should be called whenever the outer function’s asynchronous process is done. Usually, the result from outer function is passed to the callback. One example from fs.readFile
(I created a file named a.txt with content “Hello World”):
1 | var fs = require("fs"); |
Output:
1 | content: undefined |
The callback passed to fs.readFile
prints the content of file and assign it to content
, but given the output variable content
is not assigned. That is asynchronous: the callback is not processed immediately while other functions are processed before it. Only after fs.readFile
gets results, and callback becomes the first element in message queue, the callback can be processed.
Promise
Callback Hell is a “well-known” style in JavaScript. It is common to nest 3 or more callbacks given each callback rely on previous callback, which makes the code unreadable, especially when comming into error handling. Then people come up with Promise
that became standard in ES6. There are numerous tutorials about Promise
, here just give a single example of how Promise
performs asynchronous operation.
1 | const square = new Promise((resolve) => { |
Output:
1 | sync... Promise { 1 } |
This is a simple example to calculate 2*2. Compared to multi-leveled callbacks, it is more elegant. You can do infinite asynchronous operatios in .then()
by chaining forever. And you no longer to nest error handling in callback, while you can just use .catch()
to handle the first error encountered in .then()
chain.
Async/await
Then what is a more elegant way without .then()
chain because it might be still awkward looking? Here comes async/await
. Still there are many tutorials so that here just giving an example of how to handle asynchronous operation through async/await
1 | const num = function (val) { |
Output:
1 | 2 * 5 = 10 |
Above is a simple example of async/await
usage. In brief, async function
returns a Promise
; await
blocks execution until the promise is resolved or rejected by which it mimics synchronouse operation, and returns the resolved value, while error is catched through try...catch...
.
In addition, it is elegant to perform concurrency by Promise.all()
. You see concur()
is printed before seq()
. The reason is, although seq()
is called firstly, it needs to resolve 3 promises one by one, while concur()
resolve the 3 promises at once(not an accurate description but at lease looks like). Thus concur()
returns before seq()
.
Afterword
Promise
and async/awiat
do give us an easier way to write asychronous codes. However, it trades detailed control for elegancy. In callback you can deal with whatever you want, whereas Promise
and async/awiat
reduce many details. It might be ok when project is not too complicated, but it does have drawbacks. This is a large topic and will be discussed in future post.
Reference
The Event Loop
Event loop: microtasks and macrotasks
Is JavaScript guaranteed to be single-threaded?
What the heck event loop is (video)