Last Updated : 02 Aug, 2025
A callback function is a function that is passed as an argument to another function and executed later.
function greet(name, callback) {
console.log("Hello, " + name);
callback();
}
function sayBye() {
console.log("Goodbye!");
}
greet("Ajay", sayBye);
Hello, Ajay Goodbye!
Here, sayBye() is passed as a callback to greet(), which executes after the greeting.
How Do Callbacks Work in JavaScript?JavaScript executes code line by line (synchronously), but sometimes we need to delay execution or wait for a task to complete before running the next function. Callbacks help achieve this by passing a function that is executed later.
Callbacks for Asynchronous Execution
JavaScript
console.log("Start");
setTimeout(function () {
console.log("Inside setTimeout");
}, 2000);
console.log("End");
Output
Start
End
Inside setTimeout (after 2 seconds)
Callbacks are widely used in
When a function needs to execute different behaviors based on input, callbacks make the function flexible.
JavaScript
function calc(a, b, callback) {
return callback(a, b);
}
function add(x, y) {
return x + y;
}
function mul(x, y) {
return x * y;
}
console.log(calc(5, 3, add));
console.log(calc(5, 3, mul));
JavaScript is event-driven, and callbacks handle user interactions like clicks and key presses.
JavaScript
document.getElementById("myButton").addEventListener("click", function () {
console.log("Button clicked!");
});
Here, the anonymous function is a callback that runs when the button is clicked.
4. Callbacks in API Calls (Fetching Data)Callbacks are useful when retrieving data from APIs.
JavaScript
function fetch(callback) {
fetch("https://jsonplaceholder.typicode.com/todos/1")
.then(response => response.json())
.then(data => callback(data))
.catch(error => console.error("Error:", error));
}
function handle(data) {
console.log("Fetched Data:", data);
}
fetch(handle);
fetchData() gets data from an API and passes it to handleData() for processing.
Problems with CallbacksAlthough callbacks are useful, they have some drawbacks.
1. Callback Hell (Nested Callbacks)When callbacks are nested deeply, the code becomes unreadable and hard to maintain.
JavaScript
function step1(callback) {
setTimeout(() => {
console.log("Step 1 completed");
callback();
}, 1000);
}
function step2(callback) {
setTimeout(() => {
console.log("Step 2 completed");
callback();
}, 1000);
}
function step3(callback) {
setTimeout(() => {
console.log("Step 3 completed");
callback();
}, 1000);
}
step1(() => {
step2(() => {
step3(() => {
console.log("All steps completed");
});
});
});
As the number of steps increases, the nesting grows deeper, making the code difficult to manage.
2. Error Handling Issues in CallbacksError handling can get messy when dealing with nested callbacks.
JavaScript
function divide(a, b, callback) {
if (b === 0) {
callback(new Error("Cannot divide by zero"), null);
} else {
callback(null, a / b);
}
}
function result(error, result) {
if (error) {
console.log("Error:", error.message);
} else {
console.log("Result:", result);
}
}
divide(10, 2, result);
divide(10, 0, result);
Result: 5 Error: Cannot divide by zero
Handling errors inside callbacks can complicate code readability.
Alternatives to Callbacks 1. Promises (Fixing Callback Hell)Promises provide a better way to handle asynchronous tasks without deep nesting.
JavaScript
function step1() {
return new Promise(resolve => {
setTimeout(() => {
console.log("Step 1 completed");
resolve();
}, 1000);
});
}
function step2() {
return new Promise(resolve => {
setTimeout(() => {
console.log("Step 2 completed");
resolve();
}, 1000);
});
}
function step3() {
return new Promise(resolve => {
setTimeout(() => {
console.log("Step 3 completed");
resolve();
}, 1000);
});
}
step1()
.then(step2)
.then(step3)
.then(() => console.log("All steps completed"));
Promises make code more readable by chaining .then() instead of nesting callbacks.
2. Async/Await (Cleaner Alternative)async/await provides an even cleaner way to handle asynchronous code.
JavaScript
async function processSteps() {
await step1();
await step2();
await step3();
console.log("All steps completed");
}
processSteps();
async/await makes code look synchronous, improving readability.
When to Use and Avoid Callbacks?Use callbacks when
Avoid callbacks when
RetroSearch is an open source project built by @garambo | Open a GitHub Issue
Search and Browse the WWW like it's 1997 | Search results from DuckDuckGo
HTML:
3.2
| Encoding:
UTF-8
| Version:
0.7.4