javascript
async function promiseThrow() {
return new Promise((resolve, reject) => {
throw new Error("promise error");
});
}
//这样会正常处理
async function promiseThrow() {
return new Promise(async (resolve, reject) => {
throw new Error("promise error");
});
}
//这样会出错,外部无法捕获
1. 不使用 async
关键字的 Promise 构造函数回调函数
ts
async function promiseThrow() {
return new Promise((resolve, reject) => { throw new Error("promise error"); });
}
promiseThrow().catch((error) => { console.error("Caught error:", error.message); });
在这个例子中,Promise 构造函数的回调函数内部直接 throw new Error("promise error")
。虽然这是一个同步错误,但由于它发生在 Promise 构造函数的执行环境中,JavaScript 引擎会将这个错误封装成一个 rejected Promise,并将其关联到返回的 Promise 实例上。因此,外部通过 .catch
方法可以正常捕获这个错误。 2. 使用 async
关键字的 Promise 构造函数回调函数
ts
async function promiseThrow() {
return new Promise(async (resolve, reject) => { throw new Error("promise error"); });
}
promiseThrow().catch((error) => { console.error("Caught error:", error.message); });
在这个例子中,Promise 构造函数的回调函数是一个 async
函数。当 throw new Error("promise error")
执行时,它会生成一个 rejected Promise,但这个 rejected Promise 并未被 Promise 构造函数捕获和处理。由于 async
函数内部的错误被转换为 rejected Promise,这个 rejected Promise 只会在 async
函数返回的 Promise 中体现出来,而 Promise 构造函数本身并未等待这个内部 async
函数的结果。因此,外部的 .catch
无法捕获到这个错误,导致程序意外终止。
结论:
- 当在不使用
async
关键字的 Promise 构造函数回调函数中throw
错误时,错误会被 Promise 机制捕获并转换为 rejected Promise,外部的.catch
可以正常捕获这个错误。 - 当在使用
async
关键字的 Promise 构造函数回调函数中throw
错误时,由于async
函数返回的 Promise 被忽略了,错误实际上未被 Promise 构造函数捕获,导致外部的.catch
无法捕获,程序意外终止。
对于这种情况,正确的做法是避免在 Promise 构造函数回调函数中使用 async
关键字,或者在 async
回调函数内部使用 reject
来传递错误,如下所示:
ts
async function promiseThrow() {
return new Promise((resolve, reject) => { reject(new Error("promise error")); });
}
// 或者
async function promiseThrow() {
return new Promise((resolve, reject) => { const innerPromise = async () => { throw new Error("promise error");
};
innerPromise().then(resolve, reject); }); }
// 外部捕获错误
promiseThrow().catch((error) => { console.error("Caught error:", error.message); });