Spring Boot 全局异常处理(上)

背景

对接的项目多了,奇奇怪怪的问题就都出现了,比如有一个最让人烦心的问题 异常

偶尔会碰到框架抛出的默认的异常,比如 Laraval,比如 Spring Boot,每个框架抛出的异常格式是不一致的,有 Json 或 XML 格式的数据,更甚至有 HTML 页面,最为关键的是响应的数据结构和接口约定的数据结构不一致,所以这时候我们在对响应内容进行解析的时候反而会给我们自己的代码带来需要处理的异常。

基于此,为了对自己的接口负责,我们需要进行全局的异常处理,目的是防止出现约定之外的数据结构。

Spring Boot 默认的异常处理机制

默认情况下,Spring Boot 会返回两种类型的异常,一种是 HTML,还有一种是 Json 格式的数据,这主要取决于请求头中的 Accept 参数,比如浏览器发出的请求,请求头中会附带 Accept:text/html,所以此时 Spring Boot 会返回一个错误页面,称为 Whitelabel Error Page,而当我们使用 Postman 请求时,返回的则是 Json 类型的数据。

原理其实也很简单,Spring Boot 默认提供了程序出错的结果映射路径 /error。而这个 /error 请求会由 BasicErrorController 来处理,其内部其实就是通过判断请求头的 Accept 中的内容来进行区分处理逻辑的(判断是否包含 text/html),从而来决定返回页面视图还是 JSON 消息内容。

相关 BasicErrorController 中代码如下:

BasicErrorController 处理逻辑

自定义错误页面

自定义错误页面的好处有好多,比如 404 错误页面,我们完全可以自定义 404 的 HTML 页面,上面可以放置图片等,这样体验就更友好一点。

自定义的错误页面有两种,一种是 静态页面,一种是使用 模板引擎 动态生成,后者的优势是可以在页面上显示自定义的内容。

  • 静态页面的方式,html 文件的路径为:resources/public/error/xxx.html

如果要替换 404 错误页面,则在此路径下放置 404.html 文件,同理,如果要替换 500 错误页面,则在此路径下放置 500.html 文件即可

  • 模板引擎渲染的动态页面的方式,html 文件的路径为:resources/templates/error/xxx.html

文件命名同上

注意:动态页面的优先级是要高于静态页面的
比如你同时配置了静态页面和动态页面,那么最终生效的,会是动态页面。

附上文件结构图:

文件结构

自定义错误信息

上面介绍了最简单的错误处理,最主要的针对返回的 HTML,但是我们往往也要处理 Json 类型的返回内容,目的是让数据结构和我们的接口返回的数据结构一致。

Step 1 自定义 servlet 容器

需要注意的是,Spring Boot 2.x 和 Sprig Boot 1.x 是不一样的。

此处 Demo 我们仅处理了 404 和 500 这两种异常。

@Configuration
public class ContainerConfig {

    /** 下面是 springboot 2.x 系列的写法 */
    @Bean
    public WebServerFactoryCustomizer<ConfigurableWebServerFactory> webServerFactoryCustomizer(){
        return factory -> {
            factory.addErrorPages(new ErrorPage(HttpStatus.INTERNAL_SERVER_ERROR, "/error/500"));
            factory.addErrorPages(new ErrorPage(HttpStatus.NOT_FOUND, "/error/404"));
        };
    }

    /** 下面是 springboot 1.x 系列的写法 */
    /*@Bean
    public EmbeddedServletContainerCustomizer containerCustomizer(){
        return new EmbeddedServletContainerCustomizer(){
            @Override
            public void customize(ConfigurableEmbeddedServletContainer container) {
                container.addErrorPages(new ErrorPage(HttpStatus.INTERNAL_SERVER_ERROR, "/error/500"));
                container.addErrorPages(new ErrorPage(HttpStatus.NOT_FOUND, "/error/404"));
            }
        };
    }*/
}

Step 2 自定义对应的请求处理类

@Controller
public class MyBasicErrorController extends BasicErrorController {

    public MyBasicErrorController() {
        super(new DefaultErrorAttributes(), new ErrorProperties());
    }

    /**
    * @Description: 定义500的ModelAndView
    * @Param: [request, response]
    * @return: org.springframework.web.servlet.ModelAndView 
    * @Author: Jet.Chen
    * @Date: 2019-07-17 21:56
    */
    @RequestMapping(produces = "text/html",value = "/500")
    public ModelAndView errorHtml500(HttpServletRequest request, HttpServletResponse response) {
        response.setStatus(getStatus(request).value());
        Map<String, Object> model = getErrorAttributes(request, isIncludeStackTrace(request, MediaType.TEXT_HTML));
        model.put("msg","自定义错误信息");
        return new ModelAndView("error/500", model);
    }


    /**
    * @Description: 定义500 和 404 的错误JSON信息
    * @Param: [request]
    * @return: org.springframework.http.ResponseEntity<cn.jetchen.steecrserver.config.STCRResposeData>
    * STCRResposeData 为全局统一的接口数据结构
    * @Author: Jet.Chen
    * @Date: 2019-07-17 23:13
    */
    @RequestMapping(value = {"/500", "/404"})
    @ResponseBody
    public ResponseEntity<STCRResposeData> error500(HttpServletRequest request) {
        Map<String, Object> body = getErrorAttributes(request, isIncludeStackTrace(request, MediaType.TEXT_HTML));
        HttpStatus status = getStatus(request);
        Object messageTemp;
        // stcrResposeData 为返回的数据
        STCRResposeData stcrResposeData = STCRResposeData.initError(
                String.format("%d%d", 1, status.value()), 
                (messageTemp = body.get("error")) == null ? null : messageTemp.toString(),
                new HashMap<String, Object>() {{
                    put("error", body.get("error"));
                    put("message", body.get("message"));
                }});
        return new ResponseEntity<>(stcrResposeData, status);
    }


    /**
    * @Description: 定义404的ModelAndView
    * @Param: [request, response]
    * @return: org.springframework.web.servlet.ModelAndView 
    * @Author: Jet.Chen
    * @Date: 2019-07-17 23:13
    */
    @RequestMapping(produces = "text/html",value = "/404")
    public ModelAndView errorHtml400(HttpServletRequest request, HttpServletResponse response) {
        response.setStatus(getStatus(request).value());
        Map<String, Object> model = getErrorAttributes(request, isIncludeStackTrace(request, MediaType.TEXT_HTML));
        model.put("msg","自定义错误信息");
        return new ModelAndView("error/404", model);
    }
}

小结

全局异常的处理是非常有必要的,但是此文到此的处理方式其实才完成了一半,这些会在下篇文章中介绍,主要是因为此处处理的是全局的错误,是在过滤器之外的,但是我们希望处理的粒度更细一点。

比如在控制器层的全局处理方式:@ControllerAdvice,从抽象概念上可以理解成它是处理那些在 Controller 方法中抛出的异常。

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

推荐阅读更多精彩内容