Promise.all 中的错误处理

2022-08-29 23:33:15

我有一系列承诺,我正在解决Promise.all(arrayOfPromises);

我继续承诺链。看起来像这样

existingPromiseChain = existingPromiseChain.then(function() {
  var arrayOfPromises = state.routes.map(function(route){
    return route.handler.promiseHandler();
  });
  return Promise.all(arrayOfPromises)
});

existingPromiseChain = existingPromiseChain.then(function(arrayResolved) {
  // do stuff with my array of resolved promises, eventually ending with a res.send();
});

我想添加一个 catch 语句来处理单个 promise,以防它出错,但是当我尝试时,它会返回它找到的第一个错误(忽略其余错误),然后我无法从数组中的其余 promise 中获取数据(没有错误)。Promise.all

我尝试过做类似的事情..

existingPromiseChain = existingPromiseChain.then(function() {
      var arrayOfPromises = state.routes.map(function(route){
        return route.handler.promiseHandler()
          .then(function(data) {
             return data;
          })
          .catch(function(err) {
             return err
          });
      });
      return Promise.all(arrayOfPromises)
    });

existingPromiseChain = existingPromiseChain.then(function(arrayResolved) {
      // do stuff with my array of resolved promises, eventually ending with a res.send();
});

但这并不能解决。

谢谢!

--

编辑:

下面的答案是完全正确的,由于其他原因,代码被破坏了。如果有人感兴趣,这就是我最终得到的解决方案...

节点快递服务器链

serverSidePromiseChain
    .then(function(AppRouter) {
        var arrayOfPromises = state.routes.map(function(route) {
            return route.async();
        });
        Promise.all(arrayOfPromises)
            .catch(function(err) {
                // log that I have an error, return the entire array;
                console.log('A promise failed to resolve', err);
                return arrayOfPromises;
            })
            .then(function(arrayOfPromises) {
                // full array of resolved promises;
            })
    };

API 调用(route.async 调用)

return async()
    .then(function(result) {
        // dispatch a success
        return result;
    })
    .catch(function(err) {
        // dispatch a failure and throw error
        throw err;
    });

将 for 放在 前面似乎已经达到了从原始承诺中捕获任何错误的目的,但随后将整个数组返回到下一个.catchPromise.all.then.then

谢谢!


答案 1

Promise.all是全有或全无。一旦阵列中的所有承诺都解析,它就会解析,或者一旦其中一个承诺被拒绝,它就会拒绝。换句话说,它要么使用包含所有已解析值的数组进行解析,要么使用单个错误进行拒绝。

有些库有一个叫做的东西,我知道它会等待数组中的所有承诺解析或拒绝,但我不熟悉它,而且它不在ES6中。Promise.when

您的代码

我同意这里的其他人的观点,即您的修复应该有效。它应该使用可能包含成功值和错误对象混合的数组进行解析。在成功路径中传递错误对象是不寻常的,但假设你的代码正在期待它们,我认为它没有问题。

我能想到为什么它会“无法解决”的唯一原因是,它没有向我们显示代码中的失败,并且您没有看到有关此的任何错误消息的原因是因为此承诺链不会以最终捕获终止(就您向我们显示的内容而言)。

我冒昧地从您的示例中分解出“现有链”,并用捕获终止链。这可能不适合你,但对于阅读本文的人来说,重要的是要始终返回或终止链,否则潜在的错误,甚至编码错误,都会被隐藏(这是我怀疑这里发生的事情):

Promise.all(state.routes.map(function(route) {
  return route.handler.promiseHandler().catch(function(err) {
    return err;
  });
}))
.then(function(arrayOfValuesOrErrors) {
  // handling of my array containing values and/or errors. 
})
.catch(function(err) {
  console.log(err.message); // some coding error in handling happened
});

答案 2

新答案

const results = await Promise.all(promises.map(p => p.catch(e => e)));
const validResults = results.filter(result => !(result instanceof Error));

未来承诺 API