我正在尝试构建一个Promises链数组并让它们按顺序执行。每个链在数组中的下一个条目执行之前完成。例如:[connect1, connect2, connect3]
其中每个“任务”包含一些包含若干步骤的链。
我遇到的一个大问题是能够构建链并将其添加到数组中 - 因为它已经开始执行。
我一直在玩的一些测试代码在这里:
function step1() {
return new Promise(resolve => {
// Normally something asnyc here
console.log("step1:", this);
resolve();
});
}
function step2() {
return new Promise(resolve => {
// Normally something asnyc here
console.log("step2:", this);
resolve();
});
}
function step3() {
return new Promise(resolve => {
// Normally something asnyc here
console.log("step3:", this);
resolve();
});
}
function promiseSeq( tasks, state ) {
let current = Promise.resolve();
for (let k = 0; k < tasks.length; ++k) {
var taskWithState = tasks[k];
if (typeof state !== 'undefined') {
taskWithState = taskWithState.bind(state);
}
current = current.then(taskWithState);
}
return current;
}
function buildChain(idx) {
// Build the connection chain (with state)
//------------------------------
var s = { conn: idx }; // some state
var q = [ step1, step2, step3 ];
return promiseSeq(q, s);
}
function streamConnect() {
// Build array of chains
//------------------------------
var q = [ ];
q.push(buildChain(1)); // e.g. connect1
q.push(buildChain(2)); // e.g. connect2
q.push(buildChain(3)); // e.g. connect3
var p = Promise.each(q, function(f) {return ( f )});
// Process promises...
p.then(function() {console.log("done")})
.catch(function(err) {console.error("catch:",err)})
return;
}
一旦我为数组中的每个“任务”提供链,我想按顺序执行链。所以我的目标是让输出为:
step1: Object {conn: 1}
step2: Object {conn: 1}
step3: Object {conn: 1}
step1: Object {conn: 2}
step2: Object {conn: 2}
step3: Object {conn: 2}
step1: Object {conn: 3}
step2: Object {conn: 3}
step3: Object {conn: 3}
而是使用我的代码我看到了:
step1: Object {conn: 1}
step1: Object {conn: 2}
step1: Object {conn: 3}
step2: Object {conn: 1}
step2: Object {conn: 2}
step2: Object {conn: 3}
step3: Object {conn: 1}
step3: Object {conn: 2}
step3: Object {conn: 3}
对于Promises我很绿,我试图理解(没有特定的顺序):
1.为什么承诺执行似乎是交错的(所有step1已完成,然后是step2等)?
2.如何获得序列化的执行行我的预期输出?
3.链条能否以延迟方式设置?我看到我的行current = current.then(taskWithState);
最终用Bluebird调用async.invoke
,但我没有办法避免这种情况。
非常感谢任何建议或帮助。
答案 0 :(得分:2)
您的buildChain()
功能执行您的操作。它立即开始运行它们。
所以,这段代码:
var q = [ ];
q.push(buildChain(1)); // e.g. connect1
q.push(buildChain(2)); // e.g. connect2
q.push(buildChain(3)); // e.g. connect3
立即开始运行所有三个链。因此,您有三个并行运行的承诺链,而不是按顺序运行。
仅供参考,这行代码应该是一个有趣的线索:
var p = Promise.each(q, function(f) {return ( f )});
因为Promise.each()
调用其回调时你实际上没有做任何事情。因此,您没有对其操作给予任何控制权。它们在Promise.each()
运行之前就已经开始了。
您可以使用Promise.each()
来解决此问题,但您需要按预期使用它。它希望你传递一个东西的数组和一个对数组中的项进行操作的迭代器。你不是那样做的。您正在向它传递一系列已经开始执行的承诺。
您可以通过将streamConnect()
更改为此来修复此问题:
function streamConnect() {
var q = [1,2,3];
var p = Promise.each(q, function(item) {
return buildChain(item);
});
// Process promises...
p.then(function() {console.log("done")})
.catch(function(err) {console.error("catch:",err)})
return;
}
这并没有开始执行Promise.each()
调用的每个链UNTIL。