一聚教程网:一个值得你收藏的教程网站

最新下载

热门教程

Spring如何进行统一的Rest异常处理 Spring进行统一Rest异常处理方法

时间:2020-05-13 编辑:袖梨 来源:一聚教程网

Spring如何进行统一的Rest异常处理?本篇文章小编给大家分享一下Spring进行统一Rest异常处理方法,小编觉得挺不错的,现在分享给大家供大家参考,有需要的小伙伴们可以来看看。

1. 前言

统一的异常处理对于应用的重要性不言而喻。今天我们来介绍一下Spring如何来进行统一的Rest异常处理。同时我们也会简单比较一下它们之间的优劣。

2. @Controller 结合 @ExceptionHandler

在控制器中声明一个方法然后用@ExceptionHandler注解标记即可:

 @Controller
 @RequestMapping("/test")
 public class TestController {
 
   @RequestMapping("/err")
   @ResponseBody
   public Object demo1(){
     int i = 1 / 0;
     return new Date();
   }
 
   @ExceptionHandler({RuntimeException.class})
   public ModelAndView fix(Exception ex){
     System.out.println(ex.getMessage());
     return new ModelAndView("error",new ModelMap("ex",ex.getMessage()));
   }
 }

优点:

优先级最高。

@ExceptionHandler标记的方法返回值类型支持多种。可以是视图,也可以是json等。

缺点:

一个Controller中的@ExceptionHandler注解上的异常类型不能出现相同的,否则运行时抛异常。

需要显式的声明处理的异常类型。

作用域仅仅是该Controller并不是真正意义上的全局异常。如果要想作用于全局需要将其放入所有控制器的父类中。

3. @ControllerAdvice 结合 @ExceptionHandler

这是2.的改进型,通过定义@ControllerAdvice类并在方法上标记@ExceptionHandler,达到了全局异常处理的目的:

 @ControllerAdvice
 public class TestController {
 
 
   @ExceptionHandler({RuntimeException.class})
   public ModelAndView fix(Exception ex){
     System.out.println(ex.getMessage());
     return new ModelAndView("error",new ModelMap("ex",ex.getMessage()));
   }
 }

优点:

全局的异常处理。

完全控制响应的主体以及状态码

将多个异常映射到同一方法,以一起处理,并且它充分利用了更新的Restful ResponseEntity响应

缺点:

一个Controller中的@ExceptionHandler注解上的异常类型不能出现相同的,否则运行时抛异常。

需要显式的声明处理的异常类型。

一般情况下也建议使用该方式进行异常处理。大多数情况下都是兼容的。

4. HandlerExceptionResolver 接口

实现HandlerExceptionResolver接口,这里我们继承其抽象实现AbstractHandlerExceptionResolver:

 @Component
 public class RestResponseStatusExceptionResolver extends AbstractHandlerExceptionResolver {
 
   @Override
   protected ModelAndView doResolveException(
    HttpServletRequest request, 
    HttpServletResponse response, 
    Object handler, 
    Exception ex) {
     try {
       if (ex instanceof IllegalArgumentException) {
         return handleIllegalArgument((IllegalArgumentException) ex, response, handler);
       }
      //todo more exception
     } catch (Exception handlerException) {
        //todo 
     }
     return null;
   }
 
   private ModelAndView 
    handleIllegalArgument(IllegalArgumentException ex, HttpServletResponse response) 
    throws IOException {
     response.sendError(HttpServletResponse.SC_CONFLICT);
     String accept = request.getHeader(HttpHeaders.ACCEPT);
      //todo more response
     return new ModelAndView();
   }
 }

优点:

这是一个全局的异常处理器。

这种方式全局异常处理返回JSP、velocity等模板视图比较方便。

支持多种格式的响应,虽然覆写的方法返回的是ModelAndView但是因为参数中有HttpServletResponse, 我们可以利用它来进行定制响应结果。例如,如果客户端要求输入apppcation / json,那么在出现错误情况时,我们要确保我们返回一个以apppcation / json编码的响应。

缺点:

我们需要与低级的HttpServletResponse交互才能实现各种形式的响应体。

优先级比较低

5. Spring Boot 中的异常处理

如果你用的框架是Spring Boot。 我们还可以用它独特的处理方式。优点是屏蔽了低级的API,缺点也比较明显,无法捕捉到具体的异常。

5.1 实现 ErrorController

Spring Boot在默认情况下,提供了/error映射来处理所有错误,在Servlet容器里注册了全局的错误页面(Whitelabel Error Page)并返回客户端。

通过实现ErrorController接口并注册为Bean。这里不再举例。可参考BasicErrorController。

5.2 添加ErrorAttributes

我们也可以添加ErrorAttributes类型的Bean来替换替换默认的异常处理。

 @Component
 public class MyCustomErrorAttributes extends DefaultErrorAttributes {
 
   @Override
   public Map getErrorAttributes(
    WebRequest webRequest, boolean includeStackTrace) {
     Map errorAttributes = 
      super.getErrorAttributes(webRequest, includeStackTrace);
     errorAttributes.put("locale", webRequest.getLocale()
       .toString());
     errorAttributes.remove("error");
 
     //todo your business
 
     return errorAttributes;
   }
 }

5.3 继承基类 BasicErrorController

Spring Boot自动配置还提供了实现ErrorController接口异常处理的基类BasicErrorController,默认是处理text/html类型请求的错误,可以继承该基类自定义处理更多的请求类型,添加公共方法并使用@RequestMapping注解的produce属性指定处理类型。

 @Component
 public class MyErrorController extends BasicErrorController {
 
   public MyErrorController(ErrorAttributes errorAttributes) {
     super(errorAttributes, new ErrorProperties());
   }
 
   @RequestMapping(produces = MediaType.APPLICATION_XML_VALUE)
   public ResponseEntity> xmlError(HttpServletRequest request) {
     
   //todo your business
 
   }
 }

6. Spring 5 的 ResponseStatusException

另外在最新的Spring 5中你还可以通过 抛出ResponseStatusException异常来进行处理。

好处:

使用比较方便

一种类型,多种状态代码:一种异常类型可以导致多种不同的响应。与@ExceptionHandler相比,这减少了紧密耦合

我们将不必创建那么多的自定义异常类

由于可以通过编程方式创建异常,因此可以更好地控制异常处理

缺点:

没有统一的异常处理方式,强制执行某些应用程序范围的约定更加困难

可能会有大量的重复代码。

热门栏目