Node.js Asynchronous Programming Callback Introduction (1)_node.js
Node.js is based on JavaScript engine v8 and is single-threaded. Node.js uses asynchronous programming methods similar to JavaScript on the Web to handle blocking I/O operations. Reading files, accessing databases, network requests, etc. in Node.js may all be asynchronous. For those new to Node.js or developers migrating to Node.js from other language backgrounds, asynchronous programming is a painful part. This chapter will introduce you to all aspects of Node.js asynchronous programming from the simplest to the deeper. From the most basic callback to thunk, Promise, co to async/await planned for ES7.
First, let’s start with a specific example of asynchronous programming.
Get weather information for multiple IP locations
In the file ip.json, there is an array where we store several IP addresses, which are different visitors from different places. The content is as follows:
// ip.json
["115.29.230.208", "180.153.132.38", "74.125.235.224", "91.239.201.98", "60.28.215.115"]
I hope I can get the current weather for each IP location. Output the results to the weather.json file in the following format:
// weather.json
[
{ "ip": "115.29.230.208", "weather": "Clouds", "region": "Zhejiang" },
{ "ip": "180.153.132.38", "weather": "Clear", "region": "Shanghai" },
{ "ip": "74.125.235.224", "weather": "Rain", "region": "California" },
{ "ip": "60.28.215.115", "weather": "Clear", "region": "Tianjin" }
]
To organize our thoughts, we divide it into the following steps:
1. Read the ip address;
2. Get the geographical location of the IP based on the IP address;
3. Check the local weather based on geographical location;
4. Write the results to the weather.json file.
These steps are all asynchronous (reading and writing files can be synchronous, but as an example, asynchronous is used).
callback
First, we try to implement it in the way that Node.js API usually provides - passing a callback as an asynchronous callback without using any library. We will use three basic modules:
1.fs: Read the IP list from the file ip.json; write the result to the file;
2.request: used to send HTTP requests, obtain geo data based on IP address, and then obtain weather data through geo data;
3.querystring: used to assemble the url parameters for sending requests.
Create a new callback.js file and introduce these modules:
// callback.js
var fs = require('fs')
var request = require('request')
var qs = require('querystring')
Read the IP list in the file, call fs.readFile to read the file content, and then use JSON.parse to parse the JSON data:
...
function readIP(path, callback) {
fs.readFile(path, function(err, data) {
If (err) {
callback(err)
} else {
Try {
data = JSON.parse(data)
callback(null, data)
} catch (error) {
callback(error)
}
}
})
}
...
The next step is to use IP to obtain geo. We use request to request an open geo service:
...
function ip2geo(ip, callback) {
var url = 'http://www.telize.com/geoip/' ip
request({
url: url,
json: true
}, function(err, resp, body) {
callback(err, body)
})
}
...
Use geo data to get weather:
...
function geo2weather(lat, lon, callback) {
var params = {
lat: lat,
lon: lon,
APPID: '9bf4d2b07c7ddeb780c5b32e636c679d'
}
var url = 'http://api.openweathermap.org/data/2.5/weather?' qs.stringify(params)
request({
url: url,
json: true,
}, function(err, resp, body) {
callback(err, body)
})
}
...
Now that we have obtained the interfaces for geo and weather, we still have a slightly more complicated problem to deal with. Because there are multiple IPs, we need to read geo in parallel and read weather data in parallel:
...
function ips2geos(ips, callback) {
var geos = []
var ip
var remain = ips.length
for (var i = 0; i < ips.length; i ) {
ip = ips[i];
(function(ip) {
ip2geo(ip, function(err, geo) {
If (err) {
callback(err)
} else {
geo.ip = ip
geos.push(geo)
remain--
}
If (remain == 0) {
callback(null, geos)
}
})
})(ip)
}
}
function geos2weathers(geos, callback) {
var weathers = []
var geo
var remain = geos.length
for (var i = 0; i < geos.length; i ) {
Geo = geos[i];
(function(geo) {
geo2weather(geo.latitude, geo.longitude, function(err, weather) {
If (err) {
callback(err)
} else {
weather.geo = geo
weather.push(weather)
remain--
}
If (remain == 0) {
callback(null, weathers)
}
})
})(geo)
}
}
...
Both ips2geos and geos2weathers use a relatively primitive method, remain to calculate the number of items waiting to be returned. If remain is 0, it means that the parallel request has ended, and the processing results are loaded into an array and returned.
The last step is to write the results into the weather.json file:
...
function writeWeather(weathers, callback) {
var output = []
var weather
for (var i = 0; i < weathers.length; i ) {
weather = weathers[i]
Output.push({
IP: weather.geo.ip,
Weather: weather.weather[0].main,
Region: weather.geo.region
})
}
fs.writeFile('./weather.json', JSON.stringify(output, null, ' '), callback)
}
...
By combining the above functions, we can achieve our goal:
...
function handlerError(err) {
console.log('error: ' err)
}
readIP('./ip.json', function(err, ips) {
if (err) {
handlerError(err)
} else {
ips2geos(ips, function(err, geos) {
if (err) {
handlerError(err)
} else {
geos2weathers(geos, function(err, weathers) {
if (err) {
handlerError(err)
} else {
writeWeather(weathers, function(err) {
if (err) {
handlerError(err)
} else {
console.log('success!')
}
})
}
})
}
})
}
})
哈哈,你妈这嵌套,你可能觉得这就是 JavaScript 异步的问题,说真的,嵌套不是 JavaScript 异步的真正问题所在。上面这段代码我们可以下面这样写:
...
function ReadIPCallback(err, ips) {
if (err) {
handlerError(err)
} else {
ips2geos(ips, ips2geosCallback)
}
}
function ips2geosCallback(err, geos) {
if (err) {
handlerError(err)
} else {
geos2weathers(geos, geos2weathersCallback)
}
}
function geos2weathersCallback(err, weathers) {
if (err) {
handlerError(err)
} else {
writeWeather(weathers, writeWeatherCallback)
}
}
function writeWeatherCallback(err) {
if (err) {
handlerError(err)
} else {
console.log('success!')
}
}
readIP('./ip.json', ReadIPCallback)
好了,这是我们 callback.js 的全部内容。运行:
node callback.js
将会生成 weater.json 文件:
[
{
"ip": "180.153.132.38",
"weather": "Clear",
"region": "Shanghai"
},
{
"ip": "91.239.201.98",
"weather": "Clouds"
},
{
"ip": "60.28.215.115",
"weather": "Clear",
"region": "Tianjin"
},
{
"ip": "74.125.235.224",
"weather": "Clouds",
"region": "California"
},
{
"ip": "115.29.230.208",
"weather": "Clear",
"region": "Zhejiang"
}
]
Then what’s the real problem?
Of course it is a problem of asynchronous. Asynchronous essentially has to deal with three things:
1. When the asynchronous operation ends, it needs to be notified back. Callback is a solution;
2. The results generated asynchronously need to be passed back. Callback accepts a data parameter and passes the data back;
3. What should I do if an asynchronous error occurs? Callback accepts an err parameter and returns the error.
But have you found a lot of repetitive work (various callbacks)? Is there anything wrong with the above code? Please look forward to the continuation of this article.

Hot AI Tools

Undresser.AI Undress
AI-powered app for creating realistic nude photos

AI Clothes Remover
Online AI tool for removing clothes from photos.

Undress AI Tool
Undress images for free

Clothoff.io
AI clothes remover

AI Hentai Generator
Generate AI Hentai for free.

Hot Article

Hot Tools

Notepad++7.3.1
Easy-to-use and free code editor

SublimeText3 Chinese version
Chinese version, very easy to use

Zend Studio 13.0.1
Powerful PHP integrated development environment

Dreamweaver CS6
Visual web development tools

SublimeText3 Mac version
God-level code editing software (SublimeText3)

Hot Topics

Summary: Asynchronous programming in C++ allows multitasking without waiting for time-consuming operations. Use function pointers to create pointers to functions. The callback function is called when the asynchronous operation completes. Libraries such as boost::asio provide asynchronous programming support. The practical case demonstrates how to use function pointers and boost::asio to implement asynchronous network requests.

JavaScript Function Asynchronous Programming: Essential Skills for Handling Complex Tasks Introduction: In modern front-end development, handling complex tasks has become an indispensable part. JavaScript function asynchronous programming skills are the key to solving these complex tasks. This article will introduce the basic concepts and common practical methods of JavaScript function asynchronous programming, and provide specific code examples to help readers better understand and use these techniques. 1. Basic concepts of asynchronous programming In traditional synchronous programming, the code is

As web applications become more complex, programmers are forced to adopt asynchronous programming to handle large numbers of requests and I/O operations. PHP: HypertextPreprocessor is no exception. To meet this need, ReactPHP has become one of the most popular asynchronous programming frameworks for PHP. In this article, we will discuss how to do asynchronous programming in PHP using ReactPHP. 1. Introduction to ReactPHP ReactPHP is an event-driven programming

How to implement asynchronous message processing in PHP Introduction: In modern web applications, asynchronous message processing is becoming more and more important. Asynchronous message processing can improve the performance and scalability of the system and improve the user experience. As a commonly used server-side programming language, PHP can also implement asynchronous message processing through some technologies. In this article, we will introduce some methods of implementing asynchronous message processing in PHP and provide code examples. Using Message Queuing Message Queuing is a way of decoupling system components, allowing different components to

In-depth understanding of the new features of PHP8: How to use asynchronous programming and code efficiently? PHP8 is the latest major version of the PHP programming language, bringing many exciting new features and improvements. One of the most prominent features is support for asynchronous programming. Asynchronous programming allows us to improve performance and responsiveness when dealing with concurrent tasks. This article will take an in-depth look at PHP8’s asynchronous programming features and introduce how to use them efficiently. First, let’s understand what asynchronous programming is. In the traditional synchronous programming model, code follows a linear sequence

3 common problems and solutions in asynchronous programming in Java frameworks: Callback Hell: Use Promise or CompletableFuture to manage callbacks in a more intuitive style. Resource contention: Use synchronization primitives (such as locks) to protect shared resources, and consider using thread-safe collections (such as ConcurrentHashMap). Unhandled exceptions: Explicitly handle exceptions in tasks and use an exception handling framework (such as CompletableFuture.exceptionally()) to handle exceptions.

The Go framework uses Go's concurrency and asynchronous features to provide a mechanism for efficiently handling concurrent and asynchronous tasks: 1. Concurrency is achieved through Goroutine, allowing multiple tasks to be executed at the same time; 2. Asynchronous programming is implemented through channels, which can be executed without blocking the main thread. Task; 3. Suitable for practical scenarios, such as concurrent processing of HTTP requests, asynchronous acquisition of database data, etc.

Introduction to JS-Torch JS-Torch is a deep learning JavaScript library whose syntax is very similar to PyTorch. It contains a fully functional tensor object (can be used with tracked gradients), deep learning layers and functions, and an automatic differentiation engine. JS-Torch is suitable for deep learning research in JavaScript and provides many convenient tools and functions to accelerate deep learning development. Image PyTorch is an open source deep learning framework developed and maintained by Meta's research team. It provides a rich set of tools and libraries for building and training neural network models. PyTorch is designed to be simple, flexible and easy to use, and its dynamic computation graph features make
