前端常见异常及如何捕获

全局捕获-架构侧

  1. js全局异常捕获:window.onEerror 和 window.addEventLisenter('error')

  2. Promise全局异常捕获:unhandledrejection

  3. 框架级的全局异常捕获

    1. React:ReactErrorBoundary
    scala 复制代码
    // ErrorBoundary的示例
    class ErrorBoundary extends React.Component {
        constructor(props) {
          super(props);
          this.state = { hasError: false };
        }
      
        componentDidCatch(error, info) {
          this.setState({ hasError: true });
          // 在这里可以做异常的上报
          logErrorToMyService(error, info);
        }
      
        render() {
          if (this.state.hasError) {
            return <h1>Something went wrong.</h1>;
          }
          return this.props.children;
      }
      
      <ErrorBoundary>
        <MyWidget />
      </ErrorBoundary>
    1. Vue:VueVue.config.errorHandler
    2. axios: 请求统一底层封装异常处理拦截器 interceptors

全局捕获-框架层

  1. Vue 用 errorHandlerReact 用 componentDidCatch

局部捕获-业务侧

可疑代码块try-catch

try catch能捕获捉到运行时非异步错误,无法捕获语法错误和异步错误。这个需要注意。

合理使用,不要过度使用。有得异常需要抛出去给外部处理。

常见的需要注意用try-catch包裹,捕获异常的情况

  1. JSON处理必须使用try catch捕获异常
javascript 复制代码
try {
  const res=fetch(*)
  JSON.parse(res); // res 为服务端返回的数据
} catch(e) {
   // 捕获到详细的错误,在这里处理日志上报或其他异常处理等逻辑,如是否提示用户,是否有异常下的兜底数据,比如使用缓存数据等
  console.error("服务端数据格式返回异常,无法解析", res);
  
}

// 注意:下面的异常try catch无法捕获

try {
  setTimeout(() => {
    undefined.map(v => v);
  }, 1000)
} catch(e) {
  console.log("捕获到异常:", e);
}
  1. async await异步请求
  2. 正则表达式处理
  3. buffer处理

Promise异常:

注意未被捕获的promise异常会作为全局异常抛出。

  1. 局部Promise捕获两种方式:
javascript 复制代码
// 1. Promise().catch()
let promise = new Promise((resolve,reject)=>{}).catch(e=>{
   // handle error
})

// 2. async/await + try/catch
let promise = new Promise();
async function test() {
  try {
    await promise;
  } catch (e) {
    // handle error
  }
}
  1. 用Promise().catch(),
  2. 全局Promise异常用window.addEventListener("unhandledrejection")

注意:

  1. Promise自己的异常只能被自己catch, 或在try/catch里以await的方式调用来捕获。否则就会作为ERR_UNHANDLED_REJECTION异常抛出到全局。
  2. 外层Promise不能不能捕获内层Promise的异常。
javascript 复制代码
let p1 = new Promise(async (resolve, reject) => {
  return reject(100); // 被捕获
});
async function fn() {
  try {
    let result = await p1;
    console.log(2, result); //这里不会执行
  } catch (e) {
    console.log("e:", e); //这里不会执行
  }
}
fn();
javascript 复制代码
let p1 = new Promise(async (resolve, reject) => {
  return reject(100); // 未被捕获,会抛出全局异常:ERR_UNHANDLED_REJECTION
});
function fn() {
  try {
    let result = p1;
    console.log(2, result); //这里不会执行
  } catch (e) {
    console.log("e:", e); //这里不会执行
  }
}
fn();
javascript 复制代码
let p1 = new Promise(async (resolve, reject) => {
  console.log("after reject");
  return Promise.reject(100); // 未被捕获,会抛出全局异常:ERR_UNHANDLED_REJECTION
});
async function fn() {
  try {
    let result = await p1;
    console.log(2, result); //这里不会执行
  } catch (e) {
    console.log("e:", e); //这里不会执行
  }
}
fn();
javascript 复制代码
let p1 = new Promise(async (resolve, reject) => {
  return Promise.reject(100); // 未被捕获,会抛出全局异常:ERR_UNHANDLED_REJECTION
}).catch((e) => {
  console.log("promise out e", e); // 这里不会执行
});
async function fn() {
  try {
    let result = await p1;
    console.log(2, result); //这里不会执行
  } catch (e) {
    console.log("e:", e); //这里不会执行
  }
}
fn();
  1. Promise里同步抛出的异常,会触发Promise.reject而被捕获。但异步抛出的异常,不会触发Promise.reject,因此不会被捕获。
javascript 复制代码
new Promise((resolve, reject) => {
  throw new Error("Error1"); // 等效于reject
}).catch((e) => {
  console.log("异常被捕获到了1");
});

new Promise(async (resolve, reject) => {
  reject(new Error("Error2"));
}).catch((e) => {
  console.log("异常被捕获到了2");
});

new Promise(async () => {
  throw new Error("Error3");
}).catch((e) => {
  console.log("异常被捕获到了3");
});

接口请求异常:

  1. Fetch 用async await + try catch
  2. Axios 请求,自行处理自定义的异常上报

全局静态资源异常监控:

window.addEventListener("error") 或者资源标签的onError属性

相关推荐
jiet_h1 分钟前
后端 Verticle 架构实战:用 NeonBeeDeployable 推送一条通知
架构
程序猿追6 分钟前
CANN ops-math仓库解读 数学算子的底层支撑与高性能实现
人工智能·架构
芷栀夏18 分钟前
从 CANN 开源项目看现代爬虫架构的演进:轻量、智能与统一
人工智能·爬虫·架构·开源·cann
程序猿追1 小时前
深度剖析 CANN ops-nn 算子库:架构设计、演进与代码实现逻辑
人工智能·架构
程序猿追1 小时前
深度解码昇腾 AI 算力引擎:CANN Runtime 核心架构与技术演进
人工智能·架构
晚霞的不甘1 小时前
CANN 编译器深度解析:TBE 自定义算子开发实战
人工智能·架构·开源·音视频
程序猿追2 小时前
昇腾算力之锚:深度解读 CANN ascend-toolkit 异构计算架构与工程实践
架构
一枕眠秋雨>o<2 小时前
深入 CANN ops-nn:昇腾 NPU 算子开发的工程化实践与架构哲学
架构
未来龙皇小蓝2 小时前
RBAC前端架构-01:项目初始化
前端·架构
island13142 小时前
CANN Catlass 算子模板库深度解析:高性能 GEMM 架构、模板元编程与融合算子的显存管理策略
人工智能·神经网络·架构·智能路由器