Abdul Ahad Siddiqui
A journey through the world of asynchronous programming with JavaScript and TypeScript
Imagine you're at your favorite coffee shop, waiting for your order. The barista is making several beverages at once, ensuring that every customer gets their drink in the shortest time possible. That's a perfect analogy for asynchronous programming, where multiple tasks are being processed at the same time without waiting for each one to finish.
In this blog post, we'll explore the magical world of asynchronous programming in JavaScript and TypeScript. Grab your favorite drink and buckle up for an exciting adventure!
In JavaScript, synchronous code runs in sequence, like a queue at the grocery store checkout. Each task has to wait for the previous one to finish before it can start.
console.log("1. Start making coffee");
console.log("2. Finish making coffee");
console.log("3. Serve coffee to customer");
The problem with this approach is that it can lead to a slow and unresponsive application, especially when dealing with time-consuming operations like fetching data from an API.
Callbacks are the first step in our journey to the async wonderland. Imagine you're at a restaurant and you order a pizza. Instead of waiting for it to be ready, you continue chatting with your friends. When the pizza is done, the waiter brings it to your table. This is similar to how callbacks work.
Here's an example using the
setTimeout
function:console.log("1. Order pizza");
setTimeout(function onPizzaReady() {
console.log("3. Serve pizza");
}, 2000);
console.log("2. Continue chatting");
Callbacks have their own issues, like the infamous "callback hell" when dealing with nested callbacks. Fortunately, there's a hero on the horizon!
Promises are like a knight in shining armor, rescuing us from callback hell. When you ask someone to promise you something, they'll either fulfill it or reject it. Similarly, Promises in JavaScript have two states: resolved or rejected.
function makePizza() {
return new Promise((resolve, reject) => {
setTimeout(() => {
resolve("Pizza is ready!");
}, 2000);
});
}
console.log("1. Order pizza");
makePizza().then(result => {
console.log("3. " + result);
});
console.log("2. Continue chatting");
Promises make chaining multiple asynchronous operations a breeze. Just keep adding
.then()
for each subsequent operation!Async/await is the pièce de résistance of our async journey. It makes our code look synchronous while still maintaining its asynchronous nature.
Imagine you're at a traffic light. The light turns green, but instead of rushing through, you wait for the car in front of you to move.
That's how async/await works. You wait for the result of an asynchronous operation before moving on to the next line, but you don't block the entire program.
Here's our pizza example using async/await:
function makePizza() {
return new Promise((resolve, reject) => {
setTimeout(() => {
resolve("Pizza is ready!");
}, 2000);
});
}
console.log("1. Order pizza");
async function servePizza() {
const result = await makePizza();
console.log("3. " + result);
}
servePizza();
console.log("2. Continue chatting");
Notice how our code looks synchronous, but it's still asynchronous under the hood. It's the best of both worlds!
TypeScript, a superset of JavaScript, adds static types to our code. This helps catch errors early on and improves code maintainability.
Let's add TypeScript to our async/await pizza example:
function makePizza(): Promise<string> {
return new Promise((resolve, reject) => {
setTimeout(() => {
resolve("Pizza is ready!");
}, 2000);
});
}
console.log("1. Order pizza");
async function servePizza(): Promise<void> {
const result: string = await makePizza();
console.log("3. " + result);
}
servePizza();
console.log("2. Continue chatting");
We've added type annotations to our function signatures and variables, making our code safer and easier to understand.
We've come a long way in our asynchronous adventure! From callbacks to Promises and async/await, we've explored the different ways to handle asynchronous operations in JavaScript and TypeScript.
Asynchronous programming enables us to create fast, responsive applications that can handle multiple tasks simultaneously, just like a barista juggling multiple orders at a coffee shop. And with TypeScript, we can make our code even more robust and maintainable.
So go ahead, embrace the power of asynchronous programming, and create amazing experiences for your users!
Hi there, I'm Abdul Ahad Siddiqui, a Software Engineer 🚀 from India. I'm a passionate learner who's always willing to learn and work across technologies and domains 💡.