美文网首页Spring boot
Springboot异常处理的五种方式

Springboot异常处理的五种方式

作者: 三也视界 | 来源:发表于2020-11-30 09:14 被阅读0次

    SpringBoot 框架异常处理有五种处理方式,从范围来说包括有全局异常捕获处理方式和局部异常捕获处理方式,接下来通过使用下面的后端代码一一对这五种捕获方式讲解。

    package com.controller;
    
    import org.slf4j.Logger;
    import org.slf4j.LoggerFactory;
    import org.springframework.stereotype.Controller;
    import org.springframework.ui.Model;
    import org.springframework.web.bind.annotation.RequestMapping;
    
    /**
    * @Description 抛出异常 Controller,测试异常
    * 
    */
    
    @Controller
    public class ExceptionController {
        
        private static final Logger log = LoggerFactory.getLogger(ExceptionController.class);
        
        @RequestMapping("/exceptionMethod")
        public String exceptionMethod(Model model) throws Exception {
            
            model.addAttribute("msg", "没有抛出异常");
            
            int num = 1/0;   //a处
            log.info(String.valueOf(num));
            
            return "home";
        }
    
    }
    

    上述代码将会在 a处抛出 ArithmeticException 异常。

    一、自定义异常错误页面

    相信大家有过这样的经历,在遇到异常时,SpringBoot 会自动跳到一个统一的异常页面,没错,SpringBoot 默认的已经提供了一套处理异常的机制,我们只需要自定义该错误页面就可以,所以这种方式就是自定义这个异常的错误页面。
      SpringBoot 默认的异常处理机制:一旦程序中出现了异常 SpringBoot 就会请求 /error 的 url 。在 SpringBoot 中提供了一个叫 BasicExceptionController 来处理 /error 请求,然后跳转到默认显示异常的页面来展示异常信息。接下来就是自定义异常错误页面了,方法很简单,就是在目录 src/main/resources/templates/ 下定义一个叫 error 的文件,可以是 jsp 也可以是 html 。

    在指定目录添加 error.html 页面前效果

    上图为在指定目录 src/main/resources/templates/ 添加 error.html 页面前效果。

    <!DOCTYPE html>
    <html>
    <head>
    <meta charset="UTF-8">
    <title>自定义 springboot 异常处理页面</title>
    </head>
    <body>
    Springboot BasicExceptionController  错误页面
    <br>
    <span th:text="${msg}"></span>
    </body>
    </html>
    

    在指定目录添加 error.html 页面后效果图:


    在指定目录添加 error.html 页面后效果

    注意:必须是在目录 src/main/resources/templates/ 下定义 error 的文件。

    二、使用 @ExceptionHandler 注解处理局部异常

    使用这个注解就容易了,但是只能处理使用 @ExceptionHandler 注解的方法的 Controller 的异常,对于其他 Controller 的异常就无能为力了,只能再使用同样的方法将使用 @ExceptionHandler 注解的方法写入要捕获异常的 Controller 中,所以不推荐使用。

    使用方式:在最上面的 ExceptionController 中加入使用 @ExceptionHandler 注解的方法代码,整个 ExceptionController 代码如下:

    package com.controller;
    
    import org.slf4j.Logger;
    import org.slf4j.LoggerFactory;
    import org.springframework.stereotype.Controller;
    import org.springframework.ui.Model;
    import org.springframework.web.bind.annotation.ExceptionHandler;
    import org.springframework.web.bind.annotation.RequestMapping;
    
    /**
    * @Description 抛出异常 Controller,测试异常
    * 
    */
    
    @Controller
    public class ExceptionController {
    
        private static final Logger log = LoggerFactory.getLogger(ExceptionController.class);
    
        @RequestMapping("/exceptionMethod")
        public String exceptionMethod(Model model) throws Exception {
    
            model.addAttribute("msg", "没有抛出异常");
    
            int num = 1/0;
            log.info(String.valueOf(num));
    
            return "home";
        }
    
        /**
         * 描述:捕获 ExceptionController 中的 ArithmeticException 异常
         * @param model 将Model对象注入到方法中
         * @param e 将产生异常对象注入到方法中
         * @return 指定错误页面
         */
        @ExceptionHandler(value = {ArithmeticException.class})
        public String arithmeticExceptionHandle(Model model, Exception e) {
    
            model.addAttribute("msg", "@ExceptionHandler" + e.getMessage());
            log.info(e.getMessage());
    
            return "error";
        }
    }
    

    代码说明:注解 @ExceptionHandlervalue 的值为数组,表示指定捕获的异常类型,这里表示捕获 ArithmeticException 异常,因为 a 处 抛出的是 ArithmeticException 异常,跳转的页面为统一的 error.html 页面,但描述信息不同,以用来区分是 SpringBoot 处理的异常还是我们自己的方法处理的异常,下面也是使用这个方式来区分。

    当访问 http://localhost:8080/exceptionMethod 时,跳转到下面页面,显示 @ExceptionHandler/ by zero ,表示我们使用 @ExceptionHandler 注解处理异常成功。

    @ExceptionHandler/ by zero

    三、使用 @ControllerAdvice + @ExceptionHandler 注解处理全局异常

    使用 @ControllerAdvice + @ExceptionHandler 注解能够处理全局异常,这种方式推荐使用,可以根据不同的异常对不同的异常进行处理。
      使用方式:定义一个类,使用 @ControllerAdvice 注解该类,使用 @ExceptionHandler 注解方法,这里我定义了一个 GlobalException 类表示来处理全局异常,代码如下:

    package com.controller;
    
    import org.slf4j.Logger;
    import org.slf4j.LoggerFactory;
    import org.springframework.ui.Model;
    import org.springframework.web.bind.annotation.ControllerAdvice;
    import org.springframework.web.bind.annotation.ExceptionHandler;
    
    /**
    * @Description 全局异常处理类
    */
    @ControllerAdvice
    public class GlobalException {
    
        private static final Logger log = LoggerFactory.getLogger(GlobalException.class);
    
        /**
         * 描述:捕获 ArithmeticException 异常
         * @param model 将Model对象注入到方法中
         * @param e 将产生异常对象注入到方法中
         * @return 指定错误页面
         */
        @ExceptionHandler(value = {ArithmeticException.class})
        public String arithmeticExceptionHandle(Model model, Exception e) {
    
            model.addAttribute("msg", "@ControllerAdvice + @ExceptionHandler :" + e.getMessage());
            log.info(e.getMessage());
    
            return "error";
        }
    
    }
    
    

    如果需要处理其他异常,例如 NullPointerException 异常,则只需要在 GlobalException 类中定义一个方法使用 @ExceptionHandler(value = {NullPointerException.class}) 注解该方法,在该方法内部处理异常就可以了。

    当访问 http://localhost:8080/exceptionMethod 时,跳转到下面页面,显示 @ControllerAdvice + @ExceptionHandler :/ by zero ,表示我们使用 @ControllerAdvice + @ExceptionHandler 注解处理异常成功。

    使用 @ControllerAdvice + @ExceptionHandler 注解处理异常成功

    四、配置 SimpleMappingExceptionResolver 类处理异常

    通过配置 SimpleMappingExceptionResolver 类处理异常也是全局范围的,通过将 SimpleMappingExceptionResolver 类注入到 Spring 容器。

    package com.config;
    
    import java.util.Properties;
    
    import org.springframework.context.annotation.Bean;
    import org.springframework.context.annotation.Configuration;
    import org.springframework.web.servlet.handler.SimpleMappingExceptionResolver;
    
    /**
    * @Description 配置 SimpleMappingExceptionResolver 类处理异常
    *
    */
    
    @Configuration
    public class GlobalException {
    
        @Bean
        public SimpleMappingExceptionResolver
            getSimpleMappingExceptionResolver(){
            SimpleMappingExceptionResolver resolver = new SimpleMappingExceptionResolver();
    
            Properties mappings = new Properties();
            /*
             * 参数一:异常的类型,注意必须是异常类型的全名
             * 参数二:视图名称
             */
            mappings.put("java.lang.ArithmeticException", "errors");
    
            //设置异常与视图映射信息的
            resolver.setExceptionMappings(mappings);
    
            return resolver;
        }
    }
    
    

    注意:在类上加上 @Configuration 注解,在方法上加上 @Bean 注解,方法返回值必须是 SimpleMappingExceptionResolver 。

    编写 errors.html 页面

    <!DOCTYPE html>
    <html>
    <head>
    <meta charset="UTF-8">
    <title>自定义 springboot 异常处理页面</title>
    </head>
    <body>
    配置 SimpleMappingExceptionResolver 类处理异常
    </body>
    </html>
    

    访问 http://localhost:8080/exceptionMethod 链接后抛出 ArithmeticException 异常,跳转到 errors.html 页面,效果图如下所示:

    配置 SimpleMappingExceptionResolver 类处理异常

    五、实现 HandlerExceptionResolver 接口处理异常

    通过实现 HandlerExceptionResolver 接口处理异常,第一步是编写类实现 HandlerExceptionResolver 接口。

    package com.config;
    
    import javax.servlet.http.HttpServletRequest;
    import javax.servlet.http.HttpServletResponse;
    
    import org.springframework.context.annotation.Configuration;
    import org.springframework.web.servlet.HandlerExceptionResolver;
    import org.springframework.web.servlet.ModelAndView;
    
    /**
    * @Description 实现 HandlerExceptionResolver 接口处理异常
    * 
    */
    
    @Configuration
    public class HandlerExceptionResolverImpl implements HandlerExceptionResolver {
    
        @Override
        public ModelAndView resolveException(HttpServletRequest request, HttpServletResponse response, Object handler,
                Exception ex) {
            ModelAndView modelAndView = new ModelAndView();
    
            modelAndView.addObject("msg", "实现 HandlerExceptionResolver 接口处理异常");
    
            //判断不同异常类型,做不同视图跳转
            if(ex instanceof ArithmeticException){
                modelAndView.setViewName("error");
            }
    
            return modelAndView;
        }
    
    }
    
    

    注意:在类上加上 @Configuration 注解。

    配置完后访问 http://localhost:8080/exceptionMethod 后效果:

    实现 HandlerExceptionResolver 接口处理异常

    定义404等错误页面

    在resources下新建一个resources.error文件夹,下面建一个404.html

    image

    访问工程中一个不存在的路径

    image

    自定义异常抛出

    自定义一个异常

    @Datapublic class MusicEntryNotFoundException extends RuntimeException {    private String keyWords;    public MusicEntryNotFoundException(String keyWords){        super("keyWords not exist");        this.keyWords = keyWords;    }}
    

    模拟抛出

      @GetMapping("/test")    public String test(String keyWords) {        throw new MusicEntryNotFoundException(keyWords);    }
    
    image

    但是在异常处理时我们希望做一些特殊处理,提示信息更友好。

    统一的异常处理

    ControllerAdvice  //@ControllerAdvice是一个@Component,用于定义@ExceptionHandlerpublic class ControllerExceptionHandler {
        @ExceptionHandler(MusicEntryNotFoundException.class)  //    表明其用于处理MusicEntryNotFoundException异常    
        @ResponseBody
        @ResponseStatus(HttpStatus.INTERNAL_SERVER_ERROR)
        public Map<String,Object> handlerMusicEntryNotFoundException(MusicEntryNotFoundException ex){        
            Map<String,Object> result = new HashMap<>();
            result.put("keyWords",ex.getKeyWords());
            result.put("message","springBoot 错误处理测试");
            return result;
        }
    }
    

    测试结果

    image

    相关文章

      网友评论

        本文标题:Springboot异常处理的五种方式

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