节点 - async.map如何工作?

时间:2018-05-04 10:49:04

标签: node.js async.js

在阅读async之后,我假设下面的代码会向控制台输出从http / API调用返回的所有值的总和;但它似乎在第一次http呼叫返回后立即触发,并且仅显示总数'值等于API返回的第一个值。

我对async.map的工作原理有什么误解?

var http = require('https');
const
async = require('async');

var MongoClient = require('mongodb').MongoClient;
var dbUrl = "mongodb://localhost:27017/";

var total = 0;

var tokens = [ {
    name : "tron"
}, {
    name : 'cardano'
}, {
    name : 'nucleus-vision'
}, {
    name : 'ripple'
}, {
    name : 'litecoin'
}, {
    name : 'havven'
}];

function run() {

    doStuff();
    setInterval(doStuff, 1 * 60 * 1000);
};

function doStuff() {

    total = 0;

    async.map(tokens, httpGet, function (value){
          console.log('async done ', total);
        });

}

function httpGet(token, callback) {

    var url = 'https://api.coinmarketcap.com/v1/ticker/' + token.name;
    http.get( url,
        function(res) {
            var body = '';

            res.on('data', function(chunk) {
                body += chunk;
            });

            res.on('end', function() {
                var jsonObj = JSON.parse(body);
                var price = parseFloat(jsonObj[0].price);

                total += price;

                MongoClient.connect(dbUrl, function(err, db) {
                    if (err)
                        throw err;
                    var dbo = db.db("crypto");
                    dbo.collection("tick").insertOne(jsonObj[0],
                            function(err, res) {
                                if (err)
                                    throw err;
                                db.close();

                            });
                });


                callback(price);
            });



        }).on('error', function(e) {
        console.log("Got an error: ", e);
    });

};


run();

2 个答案:

答案 0 :(得分:2)

传递给iteratee(callback)的

httpGet使用不正确。第一个参数(price)被视为错误。来自the docs

  

如果iteratee将错误传递给其回调,则会立即调用主callback(对于map函数)并显示错误。

所以

callback(price);

应该是

callback(null, price);

所以async在第一次迭代后不会停止。

答案 1 :(得分:1)

我相信这里有两个不同的问题:

  1. 正如您所知,我们不能像在同步代码中那样在异步代码中使用return语句,这就是我们使用回调的原因。节点样式的回调在function (err, result) {}形式上,其中第一个参数是错误(如果有的话),第二个参数是函数的结果(返回值)。根据{{​​3}}, 如果Async.map(coll, iteratee, callback)将停止执行 iteratee将错误传递给其回调。

    当您的iteratee - 函数正在调用其回调时:callback(price),您实际上会停止执行,因为价格会作为error参数传递。您要“返回”price变量的目的是调用回调:callback(null, price)

  2. 通常,docs - 函数用于

      

    appl [ying]给定列表中每个元素的函数,以相同的顺序返回结果列表。

    异步库的map函数也是这样,IE:迭代数组并返回结果项的数组,就像法线贴图(如下)方法一样。

    [1, 2, 3].map(function (nbr) { return nbr*2 }) // returns [2, 4, 6]

    将使用价格数组调用callback(IE,async.map的第三个参数)的结果参数,而不是价格的总和值。

    async.map(tokens, httpGet, function (error, total) { console.log(error); // prints undefined (unless there was an error) console.log(total); // prints an array of prices });

  3. 对于值的求和,我建议使用map函数,或者简单地将结果返回的值相加。