promise中reject和throw的使用

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); });
相关推荐
堕落年代8 分钟前
Vue主流的状态保存框架对比
前端·javascript·vue.js
OpenTiny社区19 分钟前
TinyVue的DatePicker 组件支持日期面板单独使用啦!
前端·vue.js
冴羽20 分钟前
Svelte 最新中文文档教程(22)—— Svelte 5 迁移指南
前端·javascript·svelte
树上有只程序猿23 分钟前
Vue3组件通信:多个实战场景,轻松玩转复杂数据流!
前端·vue.js
剪刀石头布啊31 分钟前
css属性值计算过程
前端·css
bin915336 分钟前
DeepSeek 助力 Vue3 开发:打造丝滑的表格(Table)之添加列宽调整功能,示例Table14基础固定表头示例
前端·javascript·vue.js·ecmascript·deepseek
小华同学ai38 分钟前
吊打中文合成!这款开源语音神器效果炸裂,逼真到离谱!
前端·后端·github
颜酱44 分钟前
后台系统从零搭建(三)—— 具体页面之部门管理(抽离通用的增删改查逻辑)
前端·javascript·react.js
qq_3325394544 分钟前
JavaScript性能优化实战指南
前端·javascript·性能优化
wkj0011 小时前
Vue 项目中,.env文件怎么用?
前端·javascript·vue.js