nodejs循环中的多个http请求

时间:2013-11-11 16:39:25

标签: javascript node.js

我正在尝试在节点中创建简单的feed阅读器,而我在node.js中遇到了多个请求的问题。 例如,我得到的表格如下:

urls = [
"http://url1.com/rss.xml",
"http://url2.com",
"http://url3.com"];

现在我想获取每个网址的内容。第一个想法是使用for(var i in urls),但这不是个好主意。最好的选择是异步进行,但我不知道如何制作它。

有什么想法吗?

编辑:

我收到了这段代码:

var data = [];
for(var i = 0; i<urls.length; i++){
    http.get(urls[i], function(response){
    console.log('Reponse: ', response.statusCode, ' from url: ', urls[i]);
    var body = '';
    response.on('data', function(chunk){
        body += chunk;
    });

    response.on('end', function() {
        data.push(body);
    });
}).on('error', function(e){
    console.log('Error: ', e.message);
});
}

问题是,首先是循环中每个元素的调用行“http.get ...”,然后在该事件之后调用response.on('data')并在该response.on('end')之后调用。它变得一团糟,我不知道如何处理它。

5 个答案:

答案 0 :(得分:37)

默认情况下,节点http请求是异步的。您可以在代码中按顺序启动它们,并调用一个在所有请求完成后启动的函数。您可以手动执行(计算已完成的vs已启动的请求)或使用async.js

这是无依赖方式(省略错误检查):

var http = require('http');    
var urls = ["http://www.google.com", "http://www.example.com"];
var responses = [];
var completed_requests = 0;

for (i in urls) {
    http.get(urls[i], function(res) {
        responses.push(res);
        completed_requests++;
        if (completed_requests == urls.length) {
            // All download done, process responses array
            console.log(responses);
        }
    });
}

答案 1 :(得分:30)

我知道这是一个老问题,但我认为更好的解决方案是使用JavaScripts Promise.all()

const request = require('request-promise');
const urls = ["http://www.google.com", "http://www.example.com"];
const promises = urls.map(url => request(url));
Promise.all(promises).then((data) => {
    // data = [promise1,promise2]
});

答案 2 :(得分:23)

您需要检查end(数据完成事件)是否已被称为确切的请求数...这是一个有效的例子:

var http = require('http');
var urls = ['http://adrianmejia.com/atom.xml', 'http://twitrss.me/twitter_user_to_rss/?user=amejiarosario'];
var completed_requests = 0;

urls.forEach(function(url) {
  var responses = [];
  http.get(url, function(res) {
    res.on('data', function(chunk){
      responses.push(chunk);
    });

    res.on('end', function(){
      if (completed_requests++ == urls.length - 1) {
        // All downloads are completed
        console.log('body:', responses.join());
      }      
    });
  });
})

答案 3 :(得分:0)

您可以将任何承诺库与&#34; .all&#34;实现。我使用RSVP库,它很简单。

var downloadFileList = [url:'http://stuff',dataname:'filename to download']
var ddownload = downloadFileList.map(function(id){
          var dataname = id.dataname;
          var url = id.url;
          return new RSVP.Promise(function(fulfill, reject) {
           var stream = fs.createWriteStream(dataname);
            stream.on('close', function() {
            console.log(dataname+' downloaded');
            fulfill();  
            });
          request(url).on('error', function(err) {
    console.log(err);
    reject();
  }).pipe(stream);
        });
        });      
        return new RSVP.hashSettled(ddownload);

答案 4 :(得分:0)

使用闭包可以轻松解决该问题。创建一个函数来处理请求并在循环中调用该函数。每次调用该函数时,它将具有自己的词法范围,并使用 closure ,即使循环结束,它也可以保留URL的地址。甚至响应是在中,闭包也可以处理这些问题。

const request = require("request");

function getTheUrl(data) {
    var options = {
        url: "https://jsonplaceholder.typicode.com/posts/" + data
    }
    return options
}

function consoleTheResult(url) {
    request(url, function (err, res, body) {
        console.log(url);
    });
}

for (var i = 0; i < 10; i++) {
    consoleTheResult(getTheUrl(i))
}