meteor - 在返回之前同步多个异步查询?

时间:2014-09-19 19:07:56

标签: javascript asynchronous meteor

所以我有一个Meteor方法,它应该告诉服务器向第三方API发送多个API请求,然后将这些查询的结果合并到一个数组中,然后返回给客户端。

但是,我似乎找不到让服务器等到所有API查询完成后再返回结果的方法。

代码的同步版本只是一个接一个地获取数据API调用,如下所示:

Meteor.methods({
    fetchData: function(APILinks) {
        var data = [];
        APILinks.forEach(function(APILink) {
            var items = HTTP.get(APILink).content.items;
            items.forEach(function (item) {
                data.push(item);
            });
        });
        return items;
    }
});

此同步代码有效。但是,我还没有找到一种让API请求异步的好方法。我能找到最接近解决方案的方法是重新定义方法以返回仅一个API请求的结果,然后让客户端循环遍历每个API链接并为每个API链接调用方法。但是,有没有办法将所有这些请求包装成一个很好的方法,只有在所有API请求完成后才返回

1 个答案:

答案 0 :(得分:12)

您必须使用HTTP.get的异步版本并使用Future收集结果。

我创建了一个使用setTimeout来模拟HTTP请求的简单示例,以便您了解原理,我建议您从此代码开始,并使用您的HTTP get请求替换虚拟setTimeout。 / p>

该示例是一个test服务器方法,它将许多任务(n)作为参数,然后启动n个任务,每个任务以索引秒计算其索引的平方。

// we use fibers which is a dependency of Meteor anyway
var Future = Npm.require("fibers/future");

Meteor.methods({
    test: function(n) {
        // build a range of tasks from 0 to n-1
        var range = _.range(n);
        // iterate sequentially over the range to launch tasks
        var futures = _.map(range, function(index) {
            var future = new Future();
            console.log("launching task", index);
            // simulate an asynchronous HTTP request using a setTimeout
            Meteor.setTimeout(function() {
                // sometime in the future, return the square of the task index
                future.return(index * index);
            }, index * 1000);
            // accumulate asynchronously parallel tasks
            return future;
        });
        // iterate sequentially over the tasks to resolve them
        var results = _.map(futures, function(future, index) {
            // waiting until the future has return
            var result = future.wait();
            console.log("result from task", index, "is", result);
            // accumulate results
            return result;
        });
        //
        console.log(results);
        return results;
    }
});

在浏览器控制台中输入> Meteor.call("test",3,function(error,result){console.log(result);});。这将在3秒后输出[0,1,4]

在您的服务器控制台中,这将输出:

// immediately :
launching task 0
launching task 1
launching task 2
// after 1 second :
result from task 0 is 0
// after 2 seconds :
result from task 1 is 1
// after 3 seconds :
result from task 2 is 4
[ 0, 1, 4 ]

HTTP.get异步版本详见Meteor docs:

http://docs.meteor.com/#http_call

如果您想更好地理解整个Future概念,请参阅光纤文档:

https://github.com/laverdet/node-fibers