- Node.js Tutorial
- NodeJS Home
- NodeJS Introduction
- NodeJS Setup
- NodeJS First App
- NodeJS REPL
- NodeJS Command Line
- NodeJS NPM
- NodeJS Callbacks
- NodeJS Events
- NodeJS Event-Loop
- NodeJS Event-Emitter
- NodeJS Global-Objects
- NodeJS Console
- NodeJS Process
- NodeJS Buffers
- NodeJS Streams
- Node.js File Handling
- Node.js File System
- Node.js Read/Write File
- Working with folders in Node.js
- HTTP and Networking
- Node.js HTTP Module
- Anatomy of an HTTP Transaction
- Node.js MongoDB
- MongoDB Get Started
- MongoDB Create Database
- MongoDB Create Collection
- MongoDB Insert
- MongoDB Find
- MongoDB Query
- MongoDB Sort
- MongoDB Delete
- MongoDB Update
- MongoDB Limit
- MongoDB Join
- Node.js MySQL
- MySQL Get Started
- MySQL Create Database
- MySQL Create Table
- MySQL Insert Into
- MySQL Select From
- MySQL Where
- MySQL Order By
- MySQL Delete
- MySQL Update
- MySQL Join
- Node.js Modules
- Node.js Modules
- Node.js Built-in Modules
- Node.js Utility Modules
- Node.js Web Module
- Node.js Advanced
- Node.js Debugger
- Node.js Scaling Application
- Node.js Packaging
- Node.js Express Framework
- Node.js RESTFul API
- Node.js Useful Resources
- Node.js Useful Resources
- Node.js Discussion
Node.js Callbacks
In Node.js, callbacks are functions passed as arguments to other functions and are executed after the completion of an asynchronous operation. They are a core feature of Node.js and are widely used for handling asynchronous tasks like reading files, querying databases, or making HTTP requests.
Key Features of Callbacks
- Asynchronous Execution: Enables non-blocking operations.
- Control Flow: Allows execution of code after a task completes.
- Error Handling: Handles errors through callback parameters.
Structure of a Callback
Example: Synchronous Callback
function greet(name, callback) {
console.log(`Hello, ${name}`);
callback();
}
function sayGoodbye() {
console.log('Goodbye!');
}
greet('Alice', sayGoodbye);
Output:
Hello, Alice
Goodbye!
Using Callbacks in Node.js
1. Reading Files (Asynchronous)
Node.js uses callbacks for file system operations.
const fs = require('fs');
// Asynchronous file read
fs.readFile('example.txt', 'utf8', (err, data) => {
if (err) {
console.error('Error reading file:', err);
return;
}
console.log('File content:', data);
});
Output (if file exists):
File content: [Content of example.txt]
2. Error-First Callback Pattern
Node.js follows the "error-first" convention where the first argument in a callback is the error (if any).
function fetchData(callback) {
setTimeout(() => {
const error = null;
const data = { id: 1, name: 'John Doe' };
callback(error, data);
}, 1000);
}
fetchData((err, data) => {
if (err) {
console.error('Error:', err);
return;
}
console.log('Data:', data);
});
Output:
Data: { id: 1, name: 'John Doe' }
Callback Hell
When multiple callbacks are nested within each other, it leads to callback hell, making the code difficult to read and maintain.
fs.readFile('file1.txt', 'utf8', (err, data1) => {
if (err) return console.error(err);
fs.readFile('file2.txt', 'utf8', (err, data2) => {
if (err) return console.error(err);
fs.readFile('file3.txt', 'utf8', (err, data3) => {
if (err) return console.error(err);
console.log('All files read:', data1, data2, data3);
});
});
});
Alternatives to Callbacks
To avoid callback hell and improve code readability, Node.js supports:
1. Promises
- Provide a cleaner syntax for chaining asynchronous operations.
const fsPromises = require('fs').promises;
fsPromises.readFile('file1.txt', 'utf8')
.then((data) => console.log('File content:', data))
.catch((err) => console.error('Error:', err));
2. Async/Await
- Simplifies asynchronous code with a synchronous flow.
async function readFileContent() {
try {
const data = await fsPromises.readFile('file1.txt', 'utf8');
console.log('File content:', data);
} catch (err) {
console.error('Error:', err);
}
}
readFileContent();
Summary
Callbacks are fundamental to Node.js for managing asynchronous operations. While they are powerful, they can lead to complex and unreadable code when overused. Alternatives like Promises and async/await
provide better ways to handle asynchronous tasks, making your code cleaner and more maintainable.