设计一个全局异常处理器

简介: cicada: 基于 Netty4 实现的快速、轻量级 WEB 框架;没有过多的依赖,核心 jar 包仅 30KB。

前言


cicada: 基于 Netty4 实现的快速、轻量级 WEB 框架;没有过多的依赖,核心 jar 包仅 30KB



针对这个轮子以前也写过相关的介绍,感兴趣的可以再翻回去看看:








这些都看完了相信对这个小玩意应该会有更多的想法。


效果


大家平时最常用的 MVC 框架当属 SpringMVC 了,而在搭建脚手架的时候相信全局异常处理是必不可少的。


Spring 用法


通常我们的做法如下:


传统 Spring 版本:


  • 实现一个 Spring 自带的接口,重写其中的方法,最后的异常处理便在此处。


  • 将这个类配置在 Springxml ,当做一个 bean 注册到 Spring 容器中。


public class CustomExceptionResolver implements HandlerExceptionResolver {
    @Override
    public ModelAndView resolveException(HttpServletRequest request,
            HttpServletResponse response, Object handler, Exception ex) {
  //自定义处理
}


<bean class="ssm.exception.CustomExceptionResolver"></bean> 


当然现在流行的 SpringBoot 也有对应的简化版本:


@ControllerAdvice
public class GlobalExceptionHandler {
    @ExceptionHandler(value = Exception.class)
    public Object defaultErrorHandler(HttpServletRequest req, Exception e) throws Exception {
        //自定义处理
    }
}


全部都换为注解形式,但本质上还是一样的。


都是要在容器中创建一个特殊的 bean,这个 bean 专门用于处理异常,当系统运行时出现异常,就从容器中找到该 bean,并执行其中的方法即可。


至于这个特殊的 bean 如何标识出来,无非就是实现某个特定接口或者用注解声明,也就对应了传统 SpringSpringBoot 的用法。


cicada 用法


cicada 在设计自己的全局异常处理器时也参考了 Spring 的相关设计,所以最终用法如下:


@CicadaBean
public class ExceptionHandle implements GlobalHandelException {
    private final static Logger LOGGER = LoggerBuilder.getLogger(ExceptionHandle.class);
    @Override
    public void resolveException(CicadaContext context, Exception e) {
        LOGGER.error("Exception", e);
        WorkRes workRes = new WorkRes();
        workRes.setCode("500");
        workRes.setMessage(e.getClass().getName() + "系统运行出现异常");
        context.json(workRes);
    }
}


自定义一个实现了 GlobalHandelException 接口的类,当请求出现异常时,页面和后台将会如下输出:



设计


看得出用法和 Spring 非常类似,也是需要实现一个接口 GlobalHandelException,同时使用 @CicadaBean 注解该类将他加载到 cicada 内置的 IOC 容器内。


当出现异常时则在这个 IOC 容器中找到该对象调用它的 resolveException 即可。

其中还可以通过 CicadaContext 全局上下文响应不同的输出(json/text/html)。


核心原理



简单画了下流程图,步骤如下:


  • 初始化时会找到实现了 GlobalHandelException 接口的类,将它实例化并注册到 IOC 容器中。


  • 当发生异常时从容器中获取到异常处理器的对象,执行其中的处理函数即可。


说了半天原理来看看源码是如何实现的。



在初始化 bean 时,如果是一个异常处理器则会将他单独存放(也就相当于前文说的打标识)。


其中的 GlobalHandelException 本身的定义也非常简单:



接下来是运行时:



而当出现异常时则会通过之前的保存的异常处理 bean 进行异常处理,在调用的同时将全局上下文及异常信息传递过去就齐活了。


这样就可以在这个实现类中实现我们自己的异常处理逻辑了。


总结


万一今后面试官问你们 SpringMVC 的异常处理是如何实现的?你该知道怎么回答了吧😏。


同时也可以发散一下,是否可以配置一个针对于某一个 controller 的异常处理,这样每个 controller 产生的异常可以单独处理,如果没有配置则进入全局异常;原理也差不多,感兴趣的朋友可以提个 PR 完成该 feature


项目源码:


github.com/TogetherOS/…


相关文章
|
19天前
|
存储 JavaScript 关系型数据库
鸿蒙开发:实现全局异常捕获和异常查看
如何灵活的拿到错误信息后,执行我们想要的逻辑,也是自研的一个诉求,比如全局监听到异常后,重启应用,或者上传到自己的服务器,或者可以在应用内查看等等,实现一个全局异常捕获,确实有很多的有用之处。
鸿蒙开发:实现全局异常捕获和异常查看
|
3月前
|
运维 Devops
全局异常处理
全局异常处理
34 0
|
6月前
|
程序员 编译器 C语言
【C++高阶(七)】C++异常处理的方式
【C++高阶(七)】C++异常处理的方式
|
6月前
|
程序员
项目中的全局异常是如何处理的
项目中的全局异常处理通常包括对预期异常(程序员手动抛出)和运行时异常的管理。项目已提供`BaseException`作为基础异常类,用于手动抛出异常,并通过`GlobalExceptionHandler`进行全局处理。`
56 4
|
运维 Prometheus 监控
java异常 | 处理规范、全局异常、Error处理
java异常 | 处理规范、全局异常、Error处理
|
6月前
thinkphp5.1全局异常类封装
thinkphp5.1全局异常类封装
54 0
thinkphp5.1全局异常类封装
全局统一异常处理
全局统一异常处理
60 0
全局异常处理,为代码保驾护航
全局异常处理,为代码保驾护航
41 0
定义全局异常和全局异常处理器
定义全局异常和全局异常处理器