【SpirngBoot组件(3)】全局异常捕获

个人学习笔记分享,当前能力有限,请勿贬低,菜鸟互学,大佬绕道

如有勘误,欢迎指出和讨论,本文后期也会进行修正和补充

前言

在未进行全局捕获的情况下,异常有两种处理结果

  1. 主动try...catch捕获,对每种情况进行针对性处理。缺点如下:
    • 代码冗长,需要考虑各种各样的情况,无端增加工作量;
    • 人非完人,不可能考虑完整所有情况,有部分情况甚至无从捕获
  2. 被动处理,不捕获,异常信息最终将会被直接打印到前端。缺点如下:
    • 正常用户根本看不懂异常信息,大量的异常信息对于用户而言跟乱码无异
    • 异常信息会打印出后端的信息,泄露后端数据,造成巨大安全隐患,==尤其是sql错误会打印出sql语句,直接暴露表名和字段名==
    • 前端仅能判断执行成功与否,而无法根据具体情况作出处理,降低了项目的健壮性

而进行全局异常捕获后,则可在控制器层面对所有异常进行处理,不同异常返回不同信息,与前端约定好数据格式后返回统一格式的数据。

因而,全局异常已经成为项目必要的组件之一。

1.介绍

通常借助注解@ControllerAdvice@ExceptionHandler完成,当有异常被抛至控制层时,便可对其进行统一处理,返回约定好的json格式,或者某个页面

2.集成

  1. 编写好controller层相关代码,此处不做赘述

  2. 创建Controller增强器

    @Slf4j
    @ControllerAdvice
    public class BaseExceptionHandler {
    }
    
    • @Slf4j:标记使用Slf4j日志框架,此处不做赘述
    • @ControllerAdvice:标记当前类为controller蹭强器
  3. 创建异常拦截方法,可创建多个

    • 拦截返回mav对象方法

          @ExceptionHandler({NestedServletException.class})
          @ResponseStatus(HttpStatus.OK)
          @ResponseBody
          public ModelAndView servletException(HttpServletRequest request, HttpServletResponse response, Exception exception) throws ServletException, IOException {
              ModelAndView mav = new ModelAndView();
              ReturnMsg message = ReturnMsg.FAIL;
              out(response, message);
              return mav;
          }
      
      • @ExceptionHandler:参数为拦截的异常类型,可枚举多种类型,但整个controller蹭强器下,一个异常仅被拦截处理一次(按照先后顺序)
      • @ResponseStatus:响应状态,通常用HttpStatus.OK即可
      • @ResponseBody:返回json对象
    • 拦截返回自定义对象方法

      @ExceptionHandler({Exception.class})
      @ResponseStatus(HttpStatus.OK)
      @ResponseBody
      public ReturnMsg processException(HttpServletRequest request, HttpServletResponse response, Exception exception) throws ServletException, IOException {
          // 打印异常信息至控制台,开始处理异常
          log.error("异常统一处理-Exception:" + exception.getLocalizedMessage(), exception);
          //异常默认为是操作失败
          ReturnMsg message = ReturnMsg.FAIL;
          // 检查异常的类型
          if (exception instanceof NestedServletException) {
              // 异步请求错误,已处理
          } else if (exception instanceof BaseException) {
              // 自定义类型的异常,转换为自定义异常
              message = ((BaseException) exception).asReturnMsg();
          } else {
              // 非自定义类型异常,打印错误信息至日志,封装ReturnMsg对象
              log.error(request.getRequestURI(), exception);
              message = ReturnMsg.FAIL;
          }
          //返回消息体
          return message;
      }
      
    • 数据写入response方法

          public static void out(ServletResponse response, ReturnMsg returnMsg) {
              PrintWriter out = null;
              try {
                  response.setContentType("application/json;charset=utf-8");
                  out = response.getWriter();
                  out.println(ObjectMapperFactory.getInstance().writeValueAsString(returnMsg));
              } catch (Exception e) {
                  e.printStackTrace();
              } finally {
                  if (null != out) {
                      out.flush();
                      out.close();
                  }
              }
          }
      

3.使用

  • 需要处理的异常,可以try...catch捕获后处理,也可以直接抛出自定义异常,如throw new BaseException("密码错误"),由controller蹭强器转换为约定的格式后返回前端
  • 无需处理的异常,就不管咯,controller蹭强器处理后返回前端信息,但请自行完善增强器,以处理各种不同情况
  • 异常会被打印到控制台,并移交给日志框架处理,用于线上版本查看日志

4.相关资料

  • 自定义异常及统一数据返回格式:请自行查阅相关资料,实现方案很简单易懂,本人也有做相关笔记

5.补充

  • 请慎用Preconditions及其类似工具检查数据,数据违规时抛出的异常将会和其余运行异常混淆,无法区分,要么放弃此类工具,要么自行寻找解决方案
  • 尽管全局处理了,但异常就是异常,是不正常的情况。前端数据异常请返回数据告知前端,后端问题请尽可能完善代码以规避。异常就是异常,处理了仍然是异常

BB两句

这类框架性组件尽早定下来,当前项目已经大范围使用了Preconditions。。。花了半天没有完美解决方案,只能做出让步进行兼容处理,处女座表示相当难受


作者:Echo_Ye

WX:Echo_YeZ

email :echo_yezi@qq.com

个人站点:在搭了在搭了。。。(右键 - 新建文件夹)

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,245评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,749评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,960评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,575评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,668评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,670评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,664评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,422评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,864评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,178评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,340评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,015评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,646评论 3 323
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,265评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,494评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,261评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,206评论 2 352