我一直在努力养成使用promises的习惯,但在尝试在Meteor环境中的服务器端代码中使用它们时会遇到问题。这就是问题所在:
if (Meteor.isServer) {
Meteor.startup(function () {
// code to run on server at startup
p = function(){
return new Promise(function(res,rej) {
res("asd");
});
};
p().then(function(asd){
console.log("asd is " + asd);
return "zxc"
}).then(Meteor.bindEnvironment(function(zxc){
console.log("zxc is " + zxc);
return "qwe"
})).then(function(qwe){
console.log("qwe is " + qwe);
});
});
}
已安装 mvrx:bluebird
包
代码也可在GitHub
获取预期产出:
asd is asd
zxc is zxc
qwe is qwe
实际输出:
asd is asd
zxc is zxc
qwe is undefined
删除Meteor.bindEnvironment
包装修复了问题,但我需要它才能在回调中使用Collections
那我在这里缺少什么?是不是可以这样使用Promises + Meteor还是有错误?
我实际上要完成的是具有重要部分结果但需要同步结束的并行管道。这样的事情。
if (Meteor.isServer) {
Meteor.startup(function () {
// code to run on server at startup
promises = [];
step1 = function(input){
return new Promise(function(res, rej){
console.log(input + ":Step 1");
res(input);
});
};
step2 = function(input){
return new Promise(function(res, rej){
console.log(input + ":Step 2");
res(input);
});
};
step3 = function(input){
return new Promise(function(res, rej){
console.log(input + ":Step 3");
res(input);
});
};
slowIO = function(input){
var inp = input;
return new Promise( function(res,rej){
setTimeout(function(){
console.log(inp + ":SlowIO");
res(inp);
},Math.random()*20000);
});
};
end = function(input){
return new Promise(function(res,rej){
console.log(input + ": done, commiting to database");
res()
});
};
for (var i = 0; i < 100; ++i) {
promises.push(step1("pipeline-" + i).then(step2).then(slowIO).then(step3).then(end));
};
Promise.all(promises).then(function(){
console.log("All complete")
});
});
}
答案 0 :(得分:3)
(更新:I've logged an issue on github to see if it can be resolved.)
使用此方式时,Meteor.bindEnvironment
似乎存在问题。
If it's called from outside a Fiber
it won't return it's value. 注意return
Fiber(runWithEnvironment).run()
目前一个简单的解决方案是返回Promise而不是结果:
// when passed as a callback to `Promise#then`
// allows it to resolve asynchronously
var asyncThen = function(fn){
return function(arg){
return new Promise(function(resolve, reject){
fn(arg, resolve, reject);
})
};
};
Promise.resolve("asd").then(function(asd){
console.log("asd is " + asd);
return "zxc"
}).then(
asyncThen(
Meteor.bindEnvironment(function(zxc, resolve, reject){
console.log("zxc is", zxc);
resolve("qwe");
})
)
).then(function(qwe){
console.log("qwe is " + qwe);
});
答案 1 :(得分:1)
承诺只是一种以同步方式编写异步代码的方法。
如果那是你的全部,为什么不使用Meteor.wrapAsync()
?在你的情况下,你有zxc
骑牛仔自己的纤维&amp;谁知道什么时候回来。蓝鸟是伟大的&amp;在客户端上超级快,但我认为使用Meteor为您提供的代码更清晰:
//*UNTESTED*//
asd = function() { return 'foo';};
asdSync = Meteor.wrapAsync(asd);
asdResult = asdSync();
qwe = function(input) {return input.reverse()};
qweSync = Meteor.wrapAsync(qwe);
qweResult = qweSync(asdResult); //should return 'oof'