
Async/Await
finally comes to Node V8 after quite a period of time. There are plenty of articles on Medium blogging about why they are better than promise-based asyc solutions and how you should use them. If you unfamiliar with async/await
, take your time to read these articles to wrap your head around it and then come back to this article.
In this article, I will cover some aspects you need to be aware of when working with async/await
.
Code below will be used throughout this blog.
const getProduct = product => {
return new Promise((resolve, reject) => {
setTimeout(() => {
if (product) {
resolve({
product
});
} else {
reject(new Error('product not specified'));
}
}, 1000);
});
};function getProductNames() {
return Promise.resolve(['apple', 'pear', 'mango']);
}
Await returns resolved value of a Promise
Basically, await
is followed by a promise instance and returns the resolved value upon execution.
This means you cannot chain returned value using then
like you would with Promise.
async function printProduct() {
const product = await getProduct('pineapple');product.then() // error then is not a function!
}
Let’s tweak getProduct
a bit to resolve with another promise.
...
if (product) {
resolve(new Promise((resolve, reject) => {
setTimeout(x => {
resolve({
product
});
}, 5000);
}));
}
...
Now, let’s try it again. Oops! In 5 seconds time, you still see the same error again since what await
does is like — I promise that I will wait for no matter how long I need to before it gets resolved.
Async returns a promise
This allows you to chain promise
using then
:
async function printProduct() {
return await getProduct('pineapple');
}var prod = printProduct();prod.then(x => {
console.log(x);
});
This also allows for another way of handling errors apart from try/catch
that we all know:
async function printProduct() {
console.log(afffa);
return await getProduct('pineapple');
}var prod = printProduct();prod.catch(err => {
console.log(err); // error caught!
});
Thanks for YingshanDeng’s point that you do need the await
in the above code since the return value of async
will be implicitly wrapped in Promise.resolve
.
Use for…of when you need to control the flow
Let’s call getProduct
on each of products returned by calling getProductNames
and see what will happen.
async function printProduct() {
const products = await getProductNames();
products.forEach(async x => {
const prod = await getProduct(x);
console.log(prod);
});
}
Console prints out the output below in one go after 1 second.
{ product: 'apple' }
{ product: 'pear' }
{ product: 'mango' }
This is working fine but not what we actually expect to see. Instead, we want them to happen in order one after another. The problem with using normal forEach
is that it does not wait until the previous promise is resolved before it starts to run the next one.
Given the fact that Array itself has iterator
interface and can be looped using for...of
. So the solution is to use for...of
which is designed to work with collections like Array, Map, Set, Generator
. What it does is it will call next
to move onto next element when the previous one has been resolved.
async function printProduct() {
const products = await getProductNames();
for (const r of products) {
const readContents = await getProduct(r); // await is a **MUST**
console.log(readContents);
}
}
Now we are talking and seeing each fruit is printed out in every one second.
Please be mindful that you must annotate getProduct
with keyword await
even though you don’t need its async resolved value otherwise process will not wait until getProduct
gets resolved before it loops the next iteration.
Other than running them in a series, we can also run them in parallel:
async function printProduct () {
const products = await getProductNames();const promises = products.map(p => getProduct(p));
let results = [];for (let promise of promises) {
results.push(await promise);
}
}
Alternatively, we can use Promise.all
to achieve the same behaviour.
async function printProduct () {
const products = await getProductNames();
const promises = products.map(p => getProduct(p));
let results = [];
results = await Promise.all(promises);
}
Conclusion
Async/Await
really makes our async operations much easier. Hopefully, this article makes you have a better understanding of how Async/Await
works so that you can start using it if you haven’t.