How to run multiple async functions then execute callback

后端 未结 3 804
甜味超标
甜味超标 2020-12-05 10:16

In my NodeJS code I need to make 2 or 3 API calls, and each will return some data. After all API calls are complete I want to collect all the data into a single JSON object

相关标签:
3条回答
  • 2020-12-05 10:25

    https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Promise/all

    Promise.all is now included with ES6 so you don't need any 3rd party libraries at all.

    "Promise.all waits for all fulfillments (or the first rejection)"

    I've setup a gist to demonstrate Promise.all() with refactoring itterations at: https://gist.github.com/rainabba/21bf3b741c6f9857d741b69ba8ad78b1

    I'm using an IIFE (Immediately Invoked Function Expression). If you're not familiar, you'll want to be for the example below though the gist shows how with using an IIFE. https://en.wikipedia.org/wiki/Immediately-invoked_function_expression

    TL;DR

    ( function( promises ){
        return new Promise( ( resolve, reject ) => {
            Promise.all( promises )
                .then( values => {
                    console.log("resolved all promises")
                    console.dir( values );
                    resolve( values.reduce( (sum,value) => { return sum+value }) ); //Use Array.prototype.reduce() to sum the values in the array
                })
                .catch( err => {
                    console.dir( err );
                    throw err;
                });
    
        });
    })([ 
        new Promise( ( resolve, reject ) => {
            console.log("resolving 1");
            resolve( 1 );
        }),
        new Promise( ( resolve, reject ) => {
            console.log("resolving 2");
            resolve( 2 );
        })
     ]).then( sum => { console.dir( { sum: sum } ) } )
    
    0 讨论(0)
  • 2020-12-05 10:26

    Promises give you Promise.all() (this is true for native promises as well as library ones like bluebird's).

    Update: Since Node 8, you can use util.promisify() like you would with Bluebird's .promisify()

    var requestAsync = util.promisify(request); // const util = require('util')
    var urls = ['url1', 'url2'];
    Promise.all(urls.map(requestAsync)).then(allData => {
        // All data available here in the order of the elements in the array
    });
    

    So what you can do (native):

    function requestAsync(url) {
        return new Promise(function(resolve, reject) {
            request(url, function(err, res, body) {
                if (err) { return reject(err); }
                return resolve([res, body]);
            });
        });
    }
    Promise.all([requestAsync('url1'), requestAsync('url2')])
        .then(function(allData) {
            // All data available here in the order it was called.
        });
    

    If you have bluebird, this is even simpler:

    var requestAsync = Promise.promisify(request);
    var urls = ['url1', 'url2'];
    Promise.all(urls.map(requestAsync)).then(allData => {
        // All data available here in the order of the elements in the array
    });
    
    0 讨论(0)
  • 2020-12-05 10:40

    Sounds like async.parallel() would also do the job if you'd like to use async:

    var async = require('async');
    
    async.parallel({
        one: function(parallelCb) {
            request('http://www.example1.com', function (err, res, body) {
                parallelCb(null, {err: err, res: res, body: body});
            });
        },
        two: function(parallelCb) {
            request('http://www.example2.com', function (err, res, body) {
                parallelCb(null, {err: err, res: res, body: body});
            });
        },
        three: function(parallelCb) {
            request('http://www.example3.com', function (err, res, body) {
                parallelCb(null, {err: err, res: res, body: body});
            });
        }
    }, function(err, results) {
        // results will have the results of all 3
        console.log(results.one);
        console.log(results.two);
        console.log(results.three);
    });
    
    0 讨论(0)
提交回复
热议问题