鉴于以下代码:
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
处理程序?
答案 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
});