美文网首页Java程序栈程序员IT技术篇
springboot的异常处理、自定义异常处理

springboot的异常处理、自定义异常处理

作者: 一直想上树的猪 | 来源:发表于2018-12-26 15:38 被阅读3次

    一、springboot的异常处理

    首先,说一下,Springboot支持两种方式的默认处理机制:一种是客户端的(基于接口),一种是网页的。说白了就是根据请求的时候Accept的类型去进行异常的处理,在html中,Accept的类型是text/html,而基于接口去访问的话,Accept的类型是/
    我们可以截图来看一下

    网页中的请求 接口中的请求

    然后,在这两种请求方式的基础上,我来随便写一个我后台中没有定义的一个接口(当然了,肯定会报404)


    网页404

    可以看到,在基于springboot的项目中,如果访问了一个未知的资源路径,系统会自动跳到一个Whitelabel Error Page的html页面中


    接口请求的error
    而在模拟客户端请求的时候,会返回一串json,里面展示了错误的一些状态等信息。那么,在springboot中是如何区分这两种请求方式的呢?

    二、剖析源码

    源码:BasicErrorController

    @RequestMapping(
            produces = {"text/html"}
        )
        public ModelAndView errorHtml(HttpServletRequest request, HttpServletResponse response) {
            HttpStatus status = this.getStatus(request);
            Map<String, Object> model = Collections.unmodifiableMap(this.getErrorAttributes(request, this.isIncludeStackTrace(request, MediaType.TEXT_HTML)));
            response.setStatus(status.value());
            ModelAndView modelAndView = this.resolveErrorView(request, response, status, model);
            return modelAndView == null ? new ModelAndView("error", model) : modelAndView;
        }
    
        @RequestMapping
        @ResponseBody
        public ResponseEntity<Map<String, Object>> error(HttpServletRequest request) {
            Map<String, Object> body = this.getErrorAttributes(request, this.isIncludeStackTrace(request, MediaType.ALL));
            HttpStatus status = this.getStatus(request);
            return new ResponseEntity(body, status);
        }
    

    可以看到,在BasicErrorController中,有两个重载的方法,一个是返回一个modelAndView,一个是返回一个map,而这两个异常处理的区别就是,在返回modelAndView的那个方法上边,匹配了produces ="text/html",但是在返回map的方法中则没有进行详细的指定,由此我们可以大致了解boot的默认异常处理机制


    springboot异常处理源码

    三、自定义异常

    1.出现问题自动跳转到固定页面

    我先来演示一遍效果,同样,我访问一个我系统后台没有定义的一个接口(404错误),然后通过浏览器访问


    404错误页面

    可以看到,访问一个不存在的接口资源,页面跳转到了一个我自己定义的一个页面

    <!DOCTYPE html>
    <html>
    <head>
        <meta charset="UTF-8">
        <title>404</title>
    </head>
    <body>
    404您所访问的页面不存在。。。
    </body>
    </html>
    

    那么这个是怎么出现的呢?
    我们知道在boot的项目结构中,在src->main->resources目录下是配置我们的配置文件的,在此目录下我们再新建一个文件夹resources,然后在新的文件夹下面再新建一个文件夹,命名为error,(注意目录结构),然后我们就可以在error目录下新建html,想让什么错误走什么样的html就去写,就比如我之前已经预测出我访问接口会报404,那么我就可以创建404.html,以此类推还可以新建500之类的错误

    错误处理

    2.客户端模拟访问异常处理

    那么在客户端模拟访问接口的时候,并不是返回页面,boot默认返回的是一个map,而且可读性很差,那么我们怎么办呢?请看下面

    四、自定义异常

    1、新建自定义异常

    我们在实际开发过程中,为了提高系统代码的质量,我们会新建很多自定义异常,比如我创建一个自定义异常UserNotExitException,然后为了方便阅读,我会定义一个接口,去传递一个id,出现异常的时候将id传给这个异常,我们来看代码
    自定义异常代码:

    package com.tinner.exception;
    
    import com.sun.javafx.iio.ios.IosDescriptor;
    
    public class UserNotExitException extends RuntimeException {
        public UserNotExitException( Integer id){
            super("user not exit");
            this.id = id;
        }
        private Integer id;
    
        public Integer getId() {
            return id;
        }
    
        public void setId(Integer id) {
            this.id = id;
        }
    }
    
    

    接口代码:

        @GetMapping("/getUser/{id:\\d+}")
        public User getUser(@PathVariable Integer id){
            throw new UserNotExitException(id);
        }
    

    为了方便我直接抛出了这个异常,而且我在抛出这个异常的时候将父类的构造方法重写,返回一个user not exit的消息,我们来看接口访问:

    自定义异常
    我们可以看到返回的message改变了,但是这个json的可读性还是有些差,下面我将会介绍一个自己定义的返回对象

    2、自定义接口返回对象

    我来新建一个ControllerExceptionHandler这个类,这个类中必须加入注解@ControllerAdvice,我定义一个handleUserNotExitException方法,返回一个map,传参必须传递UserNotExitException这个异常(也就是说,你想处理什么异常,就必须将这个异常传递到这个方法中),在这个异常中我提前定义了一个成员变量id,我想将这个id返回给客户端,同时告诉它一些详细信息,那么我们可以在map中put相关信息,代码如下:

    
    @ControllerAdvice
    public class ControllerExceptionHandler {
    
        @ExceptionHandler(UserNotExitException.class)
        @ResponseBody
        @ResponseStatus(HttpStatus.INTERNAL_SERVER_ERROR)
        public Map<String,Object> handleUserNotExitException(UserNotExitException ex){
            Map<String,Object> map  = new HashedMap();
            map.put("id",ex.getId());
            map.put("message",ex.getMessage());
            return map;
        }
    }
    

    我们可以看到在这个方法上边我写了三个注解,第一个注解是告诉框架这个方法是用来处理什么异常的;第二个注解是为了接口返回的;第三个注解是告诉框架在服务器发生什么错误的时候来走这个方法(HttpStatus.INTERNAL_SERVER_ERROR是服务器错误的时候500)
    然后我们重新访问一遍这个接口


    处理自定义异常返回

    相关文章

      网友评论

        本文标题:springboot的异常处理、自定义异常处理

        本文链接:https://www.haomeiwen.com/subject/dsuhlqtx.html