Asynchronous JavaScript: The Power of Promises and async/await

Adedeji Yusuf
4 min readMay 29, 2023

Introduction

Asynchronous is a crucial aspect of modern web development, allowing developers to efficiently handle time-consuming operations without blocking the execution of other tasks. Traditionally, callback functions were used for asynchronous operations, but they often resulted in callback hell and difficult-to-maintain code. Promises and async/await were introduced to tackle these issues and provide an elegant and readable approach to handling asynchronous code. In this article, we will explore the concept of asynchronous JavaScript and dive into two powerful techniques: Promises and async/await. These tools provide elegant solutions to handle asynchronous operations, making code more readable, maintainable, and efficient.

Understanding Asynchronous Operations

In JavaScript, asynchronous operations occur when a task takes time to complete, such as fetching data from a server, making API requests, or reading files. By default, JavaScript executes code synchronously, meaning it waits for each operation to finish before moving on to the next line. However, this can result in performance bottlenecks and unresponsive user interfaces when dealing with time-consuming tasks. Asynchronous allows us to initiate operations and continue executing other code without waiting for them to complete.

Introducing Promises

Promises are objects that represent the eventual completion or failure of an asynchronous operation. They provide a more structured and organized way to handle asynchronous code compared to callbacks. A Promise can be in one of three states: pending, fulfilled, or rejected. When a Promise is pending, the asynchronous operation is still ongoing. When it is fulfilled, the operation completed successfully, and if it is rejected, an error occurred. Promises provide a clean and structured way to handle asynchronous tasks by chaining methods like .then() and .catch(). Let’s see an example:

const fetchData = new Promise((resolve, reject) => {
setTimeout(() => {
resolve('Data fetched successfully!');
}, 2000);
});

fetchData
.then((data) => {
console.log("Success>>>", data);
})
.catch((error) => {
console.log("Error>>>", error);
});

In the code snippet above, we created a Promise called fetchData that simulates fetching data after a delay of 2 seconds. The resolve() function is called when the operation is successful, and the .then() method is used to handle the fulfilled state. If an error occurs, the reject() function is called, and the .catch() method handles the rejected state.

Leveraging or using async/await

While Promises provide a significant improvement over traditional callback-based approaches, they can still lead to complex and nested code structures. This is where async/await comes in. The async/await syntax was introduced in ECMAScript 2017 (ES8) to make asynchronous code even more readable and concise. The async/await is a syntactic sugar built on top of Promises that simplifies writing and managing asynchronous code. It allows developers to write asynchronous code in a synchronous manner, making it easier to read and understand. The async keyword is used to define an asynchronous function, and the await keyword is used to wait for a Promise to resolve or reject. Let’s see an example:

function fetchData() {
return new Promise((resolve, reject) => {
setTimeout(() => {
resolve('Data fetched successfully!');
}, 2000);
});
}

async function fetchDataAsync() {
try {
const data = await fetchData();
console.log(data);
} catch (error) {
console.log(error);
}
}

fetchDataAsync();

In this code snippet, we define a function fetchData() that returns a Promise. The fetchDataAsync() function is marked as async and uses the await keyword to pause the execution until the Promise is resolved or rejected. This allows us to write asynchronous code in a more linear and synchronous style.

Error Handling in async/await

Both promises and async/await offer robust error handling capabilities. With promises, you can use the .catch() method to handle errors at any point in the promise chain. Similarly, async/await provides a try-catch block to catch and handle errors.

In addition to this, async/await simplifies handling parallel asynchronous operations using promise.all(). This allows multiple promises to run simultaneously and returns a new promise that resolves when all the provided promises have resolved. Here’s an example of a simple promise.all()

const urls = ['https://test.com/1', 'https://test.com/2', 'https://test.com/3'];

function fetchData(url) {
return new Promise((resolve, reject) => {
// Simulating an asynchronous operation with setTimeout
setTimeout(() => {
resolve(`Data fetched from ${url}`);
}, 2000); // delay of 2 seconds
});
}

const promises = urls.map(url => fetchData(url));

Promise.all(promises)
.then(results => {
console.log('All promises resolved...');
results.forEach(data => {
// the response for each promise will be in data
console.log(data);
});
})
.catch(error => {
console.log('Error occurred:', error);
});

In the code snippet above, we declare an array of urls we want to fetch data from, we then have a function fetchData() that returns a promise. We mapped the array to initiate multiple asynchronous operations, each of them calling the fetchData() function. Then, we use promise.all() to wait for all the promises to resolve. It takes an array of promises as its argument. Once all the promises have resolved, the .then() callback is executed, receiving an array of results in the same order as the input promises. In the .then() callback, we iterate over the results array and log each fetched data to the console. If any of the promises reject, the .catch() callback will be triggered, allowing you to handle any errors that may occur during the parallel asynchronous operations. By using promise.all(), you can efficiently handle multiple asynchronous operations in parallel and process the results collectively once they have all resolved.

Conclusion

Yay! That’s it. Asynchronous JavaScript is a fundamental aspect of modern web development, enabling efficient code execution. Promises and async/await allow developers to handle asynchronous operations, making code more readable and maintainable. When using promises and async/await, it’s important to follow best practices such as proper error handling, avoiding excessive nesting, and leveraging parallel execution when appropriate.

If you found this article helpful, kindly share with your colleagues and friends.

Thank you for reading and Happy coding!

--

--

Adedeji Yusuf
Adedeji Yusuf

No responses yet