node.js - 在 Node.js 中的 for 循环中进行异步操作,而不使用异步库帮助程序类

标签 node.js asynchronous

我是 Node.js 的初学者。我正在尝试“learnyounode”教程中的示例。我正在尝试编写一个程序,该程序采用三个 url 参数并从这些 url 中获取一些数据,并按照提供 url 的顺序显示返回的数据。

var http = require('http');
var bl = require('bl');
var url = [];
url[0] = process.argv[2];
url[1] = process.argv[3];
url[2] = process.argv[4];
var data = [];
var remaining = url.length;
for(var i = 0; i < url.length; i++){
    http.get(url[i], function (response){
        response.setEncoding('utf8');
        response.pipe(bl(function (err, chunk){
            if(err){
                console.log(err);
            }       
            else{
                data[i] = chunk.toString(); 
                console.log(data[i]);
                remaining -= 1;
                if(remaining == 0)  {
                    for(var j = 0; j < url.length; j++){
                        console.log(data[j]);
                    }                   
                }               
            }           
        }));        
    }); 
}

我的程序中有两个console.log语句。我得到的输出如下:

It'll be chunder where lets throw a ford. We're going durry where mad as a cooee
.
Shazza got us some apples with come a strides. Mad as a swag when get a dog up y
a roo. It'll be rapt piece of piss as cunning as a trackie dacks.
As cross as a bogged with watch out for the boardies. As cunning as a digger fla
min lets get some roo bar. As dry as a piker piece of piss he hasn't got a joey.
 Lets throw a strides mate we're going digger.
undefined
undefined
undefined

看起来数据已正确获取并存储在“data”数组中,但它仍然显示未定义。 知道为什么会发生这种情况吗? 提前致谢!

最佳答案

这是 Node.js 甚至浏览器中的异步编程中非常常见的问题。您遇到的一个主要问题是,当调用异步回调时,循环变量 i 将不是您希望的那样。届时,for 循环将运行到其循环末尾,并且 i 将位于所有响应回调的最终值。

有很多方法可以解决这个问题。您可以使用闭包来close i 值,并使其对每个回调唯一可用。

var http = require('http');
var bl = require('bl');
var url = [];
url[0] = process.argv[2];
url[1] = process.argv[3];
url[2] = process.argv[4];
var data = [];
var remaining = url.length;
for(var i = 0; i < url.length; i++){
    // create closure here to uniquely capture the loop index
    // for each separate http request
    (function(index) {
        http.get(url[index], function (response){
            response.setEncoding('utf8');
            response.pipe(bl(function (err, chunk){
                if(err){
                    console.log(err);
                }       
                else{
                    data[index] = chunk.toString(); 
                    console.log(data[index]);
                    remaining -= 1;
                    if(remaining == 0)  {
                        for(var j = 0; j < url.length; j++){
                            console.log(data[j]);
                        }                   
                    }               
                }           
            }));        
        }); 
    })(i);
}

如果您经常进行 Node.js 编程,您会发现您可能想学习如何使用 Promise,因为它们对于控制异步操作​​的流程和顺序非常非常方便。

关于node.js - 在 Node.js 中的 for 循环中进行异步操作,而不使用异步库帮助程序类,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29089656/

相关文章:

Node.js 控制台异步应用程序 "best practice"

javascript - 使用 Mongoose 在 Node JS 中进行全文搜索

javascript - Node.JS 中的热交换。可能的?

node.js - 为什么XLSX Node 模块只将一半的mt输入数据转换为json

python - 如何在 Twisted Klein 中异步执行代码?

javascript - 封装在 Promise 中的异步函数与同步函数

node.js - Loopback 3中刷新AccessToken

c# - 等待 IProgress 报告方法返回

c++ - 将来尝试引用已删除的函数

ios - 将异步 Gif 加载到滚动的 UICollectionView