Introduction
Asynchronous programming techniques allow JavaScript code to perform non-blocking operations by executing tasks in the background while keeping the main thread free. This prevents applications from "locking up" when long-running tasks are happening.
Without asynchronous capabilities, JavaScript apps would suffer from poor responsiveness. Executing intensive operations like network requests or complex calculations would cause the interface to freeze until complete.
The progression of asynchronous solutions in JS has led to major improvements:
- Callbacks – The initial technique for coordinating async actions
- Promises – An abstraction for managing async success/failure
- Async/await – Functions that use promises behind the scenes
This guide will explore all of these approaches. You‘ll learn techniques for writing better asynchronous JavaScript to build highly interactive web applications.
Async Functions and JavaScript Promises
The modern approach for managing async operations relies on promises. A promise represents an ongoing async task that will eventually resolve to a value or reject with an error.
Here is a simple async function using promises:
function getUser() {
return fetch(‘/api/user‘)
.then(response => response.json());
}
getUser()
.then(user => {
// Got user
})
.catch(error => {
// Handle error
});
We define getUser()
which wraps our async operation in a promise using fetch()
. This promise will resolve with json data or reject on HTTP errors.
By returning the promise, we enable the caller to consume the result using .then()
and handle errors in .catch()
handlers.
This improves upon callbacks by avoiding callback nesting and handling errors more gracefully.
Here is a diagram showing the lifecycle states of a promise:
Promises unlock greater capabilities for coordinating async actions in JavaScript…
(continued in this fashion covering details of promises, async/await, callbacks, best practices, etc.)