Vue3如何进行全局异常处理(vue,编程语言)

时间:2024-05-01 03:46:49 作者 : 石家庄SEO 分类 : 编程语言
  • TAG :

Vue3如何进行全局异常处理

在开发组件库或者插件,经常会需要进行全局异常处理,从而实现:

  • 全局统一处理异常;

  • 为开发者提示错误信息;

  • 方案降级处理等等。

那么如何实现上面功能呢?本文先简单实现一个异常处理方法,然后结合 Vue3 源码中的实现详细介绍,最后总结实现异常处理的几个核心。

本文 Vue3 版本为 3.0.11

一、前端常见异常

对于前端来说,常见的异常比较多,比如:

  • JS 语法异常;

  • Ajax 请求异常;

  • 静态资源加载异常;

  • Promise 异常;

  • iframe 异常;

  • 等等

最常用的比如:

1. window.onerror

通过 window.onerror文档可知,当 JS 运行时发生错误(包括语法错误),触发 window.onerror()

window.onerror=function(message,source,lineno,colno,error){console.log('捕获到异常:',{message,source,lineno,colno,error});}

函数参数:

  • message:错误信息(字符串)。可用于HTML onerror=""处理程序中的 event

  • source:发生错误的脚本URL(字符串)

  • lineno:发生错误的行号(数字)

  • colno:发生错误的列号(数字)

  • error:Error对象(对象)

若该函数返回true,则阻止执行默认事件处理函数。

2. try...catch 异常处理

另外,我们也经常会使用 try...catch 语句处理异常:

try{//dosomething}catch(error){console.error(error);}

更多处理方式,可以阅读前面推荐的文章。

3. 思考

大家可以思考下,自己在业务开发过程中,是否也是经常要处理这些错误情况?那么像 Vue3 这样复杂的库,是否也是到处通过 try...catch来处理异常呢?接下来一起看看。

二、实现简单的全局异常处理

在开发插件或库时,我们可以通过 try...catch封装一个全局异常处理方法,将需要执行的方法作为参数传入,调用方只要关心调用结果,而无需知道该全局异常处理方法内部逻辑。大致使用方法如下:

consterrorHandling=(fn,args)=>{letresult;try{result=args?fn(...args):fn();}catch(error){console.error(error)}returnresult;}

测试一下:

constf1=()=>{console.log('[f1running]')thrownewError('[f1error!]')}errorHandling(f1);/*输出:[f1running]Error:[f1error!]atf1(/Users/wangpingan/leo/www/node/www/a.js:14:11)aterrorHandling(/Users/wangpingan/leo/www/node/www/a.js:4:39)atObject.<anonymous>(/Users/wangpingan/leo/www/node/www/a.js:17:1)atModule._compile(node:internal/modules/cjs/loader:1095:14)atObject.Module._extensions..js(node:internal/modules/cjs/loader:1147:10)atModule.load(node:internal/modules/cjs/loader:975:32)atFunction.Module._load(node:internal/modules/cjs/loader:822:12)atFunction.executeUserEntryPoint[asrunMain](node:internal/modules/run_main:81:12)atnode:internal/main/run_main_module:17:47*/

可以看到,当需要为方法做异常处理时,只要将该方法作为参数传入即可。但是上面示例跟实际业务开发的逻辑差得有点多,实际业务中,我们经常会遇到方法的嵌套调用,那么我们试一下:

constf1=()=>{console.log('[f1]')f2();}constf2=()=>{console.log('[f2]')f3();}constf3=()=>{console.log('[f3]')thrownewError('[f3error!]')}errorHandling(f1)/*输出:[f1running][f2running][f3running]Error:[f3error!]atf3(/Users/wangpingan/leo/www/node/www/a.js:24:11)atf2(/Users/wangpingan/leo/www/node/www/a.js:19:5)atf1(/Users/wangpingan/leo/www/node/www/a.js:14:5)aterrorHandling(/Users/wangpingan/leo/www/node/www/a.js:4:39)atObject.<anonymous>(/Users/wangpingan/leo/www/node/www/a.js:27:1)atModule._compile(node:internal/modules/cjs/loader:1095:14)atObject.Module._extensions..js(node:internal/modules/cjs/loader:1147:10)atModule.load(node:internal/modules/cjs/loader:975:32)atFunction.Module._load(node:internal/modules/cjs/loader:822:12)atFunction.executeUserEntryPoint[asrunMain](node:internal/modules/run_main:81:12)*/

这样也是没问题的。那么接下来就是在 errorHandling方法的 catch分支实现对应异常处理即可。接下来看看 Vue3 源码中是如何处理的?

三、Vue3 如何实现异常处理

理解完上面示例,接下来看看在 Vue3 源码中是如何实现异常处理的,其实现起来也是很简单。

1. 实现异常处理方法

errorHandling.ts 文件中定义了 callWithErrorHandlingcallWithAsyncErrorHandling两个处理全局异常的方法。顾名思义,这两个方法分别处理:

  • callWithErrorHandling:处理同步方法的异常;

  • callWithAsyncErrorHandling:处理异步方法的异常。

使用方式如下:

callWithAsyncErrorHandling(handler,instance,ErrorCodes.COMPONENT_EVENT_HANDLER,args)

代码实现大致如下:

//packages/runtime-core/src/errorHandling.ts//处理同步方法的异常exportfunctioncallWithErrorHandling(fn:Function,instance:ComponentInternalInstance|null,type:ErrorTypes,args?:unknown[]){letrestry{res=args?fn(...args):fn();//调用原方法}catch(err){handleError(err,instance,type)}returnres}//处理异步方法的异常exportfunctioncallWithAsyncErrorHandling(fn:Function|Function[],instance:ComponentInternalInstance|null,type:ErrorTypes,args?:unknown[]):any[]{//省略其他代码constres=callWithErrorHandling(fn,instance,type,args)if(res&&isPromise(res)){res.catch(err=>{handleError(err,instance,type)})}//省略其他代码}

callWithErrorHandling方法处理的逻辑比较简单,通过简单的 try...catch 做一层封装。而 callWithAsyncErrorHandling 方法就比较巧妙,通过将需要执行的方法传入 callWithErrorHandling方法处理,并将其结果通过 .catch方法进行处理。

2. 处理异常

在上面代码中,遇到报错的情况,都会通过 handleError()处理异常。其实现大致如下:

//packages/runtime-core/src/errorHandling.ts//异常处理方法exportfunctionhandleError(err:unknown,instance:ComponentInternalInstance|null,type:ErrorTypes,throwInDev=true){//省略其他代码logError(err,type,contextVNode,throwInDev)}functionlogError(err:unknown,type:ErrorTypes,contextVNode:VNode|null,throwInDev=true){//省略其他代码console.error(err)}

保留核心处理逻辑之后,可以看到这边处理也是相当简单,直接通过 console.error(err)输出错误内容。

3. 配置 errorHandler 自定义异常处理函数

在使用 Vue3 时,也支持指定自定义异常处理函数,来处理组件渲染函数侦听器执行期间抛出的未捕获错误。这个处理函数被调用时,可获取错误信息和相应的应用实例。文档参考:《errorHandler》使用方法如下,在项目 main.js文件中配置:

//src/main.jsapp.config.errorHandler=(err,vm,info)=>{//处理错误//`info`是Vue特定的错误信息,比如错误所在的生命周期钩子}

那么 errorHandler()是何时执行的呢?我们继续看看源码中 handleError() 的内容,可以发现:

//packages/runtime-core/src/errorHandling.tsexportfunctionhandleError(err:unknown,instance:ComponentInternalInstance|null,type:ErrorTypes,throwInDev=true){constcontextVNode=instance?instance.vnode:nullif(instance){//省略其他代码//读取errorHandler配置项constappErrorHandler=instance.appContext.config.errorHandlerif(appErrorHandler){callWithErrorHandling(appErrorHandler,null,ErrorCodes.APP_ERROR_HANDLER,[err,exposedInstance,errorInfo])return}}logError(err,type,contextVNode,throwInDev)}

通过 instance.appContext.config.errorHandler取到全局配置的自定义错误处理函数,存在时则执行,当然,这边也是通过前面定义的 callWithErrorHandling来调用。

4. 调用 errorCaptured 生命周期钩子

在使用 Vue3 的时候,也可以通过 errorCaptured生命周期钩子来捕获来自后代组件的错误。 如下:

(err:Error,instance:Component,info:string)=>?boolean

此钩子会收到三个参数:错误对象、发生错误的组件实例以及一个包含错误来源信息的字符串。此钩子可以返回 false阻止该错误继续向上传播。有兴趣的同学可以通过文档,查看具体的错误传播规则。使用方法如下,父组件监听 onErrorCaptured生命周期(示例代码使用 Vue3 setup 语法):

<template><Message></Message></template><scriptsetup>//App.vueimport{onErrorCaptured}from'vue';importMessagefrom'./components/Message.vue'onErrorCaptured(function(err,instance,info){console.log('[errorCaptured]',err,instance,info)})</script>

子组件如下:

<template><button@click="sendMessage">发送消息</button></template><scriptsetup>//Message.vueconstsendMessage=()=>{thrownewError('[testonErrorCaptured]')}</script>

当点击「发送消息」按钮,控制台便输出错误:

[errorCaptured]Error:[testonErrorCaptured]atProxy.sendMessage(Message.vue:36:15)at_createElementVNode.onClick._cache.<computed>._cache.<computed>(Message.vue:3:39)atcallWithErrorHandling(runtime-core.esm-bundler.js:6706:22)atcallWithAsyncErrorHandling(runtime-core.esm-bundler.js:6715:21)atHTMLButtonElement.invoker(runtime-dom.esm-bundler.js:350:13)Proxy{sendMessage:ƒ,…}nativeeventhandler

可以看到 onErrorCaptured生命周期钩子正常执行,并输出子组件 Message.vue内的异常。

那么这个又是如何实现呢?还是看 errorHandling.ts 中的 handleError() 方法:

//packages/runtime-core/src/errorHandling.tsexportfunctionhandleError(err:unknown,instance:ComponentInternalInstance|null,type:ErrorTypes,throwInDev=true){constcontextVNode=instance?instance.vnode:nullif(instance){letcur=instance.parent//theexposedinstanceistherenderproxytokeepitconsistentwith2.xconstexposedInstance=instance.proxy//inproductionthehookreceivesonlytheerrorcodeconsterrorInfo=__DEV__?ErrorTypeStrings[type]:typewhile(cur){consterrorCapturedHooks=cur.ec//①取出组件配置的errorCaptured生命周期方法if(errorCapturedHooks){//②循环执行errorCaptured中的每个Hookfor(leti=0;i<errorCapturedHooks.length;i++){if(errorCapturedHooks[i](err,exposedInstance,errorInfo)===false){return}}}cur=cur.parent}//省略其他代码}logError(err,type,contextVNode,throwInDev)}

这边会先获取 instance.parent作为当前处理的组件实例进行递归,每次将取出组件配置的 errorCaptured 生命周期方法的数组并循环调用其每一个钩子,然后再取出当前组件的父组件作为参数,最后继续递归调用下去。

5. 实现错误码和错误消息

Vue3 还为异常定义了错误码和错误信息,在不同的错误情况有不同的错误码和错误信息,让我们能很方便定位到发生异常的地方。错误码和错误信息如下:

//packages/runtime-core/src/errorHandling.tsexportconstenumErrorCodes{SETUP_FUNCTION,RENDER_FUNCTION,WATCH_GETTER,WATCH_CALLBACK,//...省略其他}exportconstErrorTypeStrings:Record<number|string,string>={//省略其他[LifecycleHooks.RENDER_TRACKED]:'renderTrackedhook',[LifecycleHooks.RENDER_TRIGGERED]:'renderTriggeredhook',[ErrorCodes.SETUP_FUNCTION]:'setupfunction',[ErrorCodes.RENDER_FUNCTION]:'renderfunction',//省略其他[ErrorCodes.SCHEDULER]:'schedulerflush.ThisislikelyaVueinternalsbug.'+'Pleaseopenanissueathttps://new-issue.vuejs.org/?repo=vuejs/vue-next'}

当不同错误情况,根据错误码 ErrorCodes来获取 ErrorTypeStrings错误信息进行提示:

//packages/runtime-core/src/errorHandling.tsfunctionlogError(err:unknown,type:ErrorTypes,contextVNode:VNode|null,throwInDev=true){if(__DEV__){constinfo=ErrorTypeStrings[type]warn(`Unhandlederror${info?`duringexecutionof${info}`:``}`)//省略其他}else{console.error(err)}}

6. 实现 Tree Shaking

关于 Vue3 实现 Tree Shaking 的介绍,可以看我之前写的高效实现框架和 JS 库瘦身。其中,logError 方法中就使用到了:

//packages/runtime-core/src/errorHandling.tsfunctionlogError(err:unknown,type:ErrorTypes,contextVNode:VNode|null,throwInDev=true){if(__DEV__){//省略其他}else{console.error(err)}}

当编译成 production 环境后,__DEV__分支的代码不会被打包进去,从而优化包的体积。

 </div> <div class="zixun-tj-product adv-bottom"></div> </div> </div> <div class="prve-next-news">
本文:Vue3如何进行全局异常处理的详细内容,希望对您有所帮助,信息来源于网络。
上一篇:小程序如何实现全文收起功能下一篇:

14 人围观 / 0 条评论 ↓快速评论↓

(必须)

(必须,保密)

阿狸1 阿狸2 阿狸3 阿狸4 阿狸5 阿狸6 阿狸7 阿狸8 阿狸9 阿狸10 阿狸11 阿狸12 阿狸13 阿狸14 阿狸15 阿狸16 阿狸17 阿狸18