我正在尝试构建一系列 Promise 链并让它们按顺序执行。每个链在数组中的下一个条目执行之前完成。例如:[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}
当谈到 Promise 时,我非常陌生,并且我正在尝试理解(没有特定顺序):
1. 为什么 Promise 执行看起来是交错的(所有步骤 1 都完成,然后是步骤 2,等等)?
2. 如何获得上述预期输出的序列化执行行?
3. 链可以延期设置吗?我看到我的行 current = current.then(taskWithState);
最终使用 Bluebird 调用 async.invoke
,但我没有找到避免这种情况的方法。
任何建议或帮助将不胜感激。
最佳答案
您的 buildChain()
函数执行您的操作。它立即开始运行它们。
所以,这段代码:
var q = [ ];
q.push(buildChain(1)); // e.g. connect1
q.push(buildChain(2)); // e.g. connect2
q.push(buildChain(3)); // e.g. connect3
立即开始运行所有三个链。因此,您有三个并行运行的 promise 链,而不是按顺序运行。
仅供引用,这行代码应该是一个有趣的线索:
var p = Promise.each(q, function(f) {return ( f )});
因为当 Promise.each()
调用其回调时,您实际上并没有执行任何操作。因此,您没有赋予它对您的运营的任何控制权。它们在 Promise.each()
运行之前就已经启动了。
您可以使用Promise.each()来解决这个问题,但您需要按预期使用它。它希望您向它传递一个包含某些内容的数组和一个对数组中的项目进行操作的迭代器。你没有做那样的事。您向它传递了一系列已经开始执行的 promise 。
您可以通过将 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()
调用。
关于javascript - Bluebird 中 Promise 链数组的串行执行,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36945655/