为什么没有承诺异步操作引发的捕获错误?

时间:2015-11-11 13:14:47

标签: javascript error-handling promise es6-promise

鉴于以下代码:

var fs = require('fs');

var asyncErrorPromise = new Promise(function(resolve) {
  fs.readFile('does/not/exist.blah', function(fileError, content) {
    if (fileError) throw fileError;
    resolve(content);
  });
});

asyncErrorPromise
  .then(function(content) {
    console.log('content received: ' + content);
  })
  .catch(function(err) {
    console.log('error received: ' + err.name);
  });

我希望抛出的fileError(因为我尝试加载的文件不存在)将被Promise捕获并汇集到.catch()语句中。所以最后我得到了这个输出:error received: ENOENT

但是,当我运行此代码时,我得到以下内容:

    if (err) throw err;
             ^

Error: ENOENT: no such file or directory, open 'does/not/exist.blah'
    at Error (native)

所以错误没有被捕获:它像往常一样投掷,好像它在Promise之外。

为什么会这样?

我能找到的关于Promises的所有内容都更关心的是被误吞的错误而不是这个问题,当你不想要它时,它实际上会抛出!

应该如何我在这里实现了我的意图?我是否需要使用try-catch语句和reject处理程序?

1 个答案:

答案 0 :(得分:2)

您无法捕获使用promises在异步回调函数中抛出的错误,因为它的上下文将丢失。在使用Node的情况下,您将在process.on('uncaughtException')处理程序中丢失错误的上下文,并且回调将在事件队列中作为自己的事件运行,具有自己的上下文和范围。

使用promises,正确的解决方案将是reject包装承诺。

另一种方法是使用Domain对象(Domain documentation),但它正在等待弃用。使用domain,您可以维护上下文并为错误注册处理程序,如下所示。

var domain = require('domain');

var d = domain.create();

d.on('error', function(err){
    //Do something with your error
});

d.run(function(){
    //Run your async code that might throw error
});