从零搭建后端框架——异常统一处理的三种方式

新闻 前端
三种方式都能很好对异常进行统一处理,但是一般推荐使用@ControllerAdvice + @ExceptionHandler方式, 这样能够使异常处理与业务逻辑分离,并且不用自己处理Json数据返回。

 [[327655]]

前言

项目在运行时出现异常时,如若没有对异常进行捕获并处理,就会出现如下页面:

从零搭建后端框架——异常统一处理的三种方式

这样显然对用户是极其不友好的。

后端不应该直接返回错误页面,而应返回统一的错误信息,比如:

  1. "code"500
  2. "data"null
  3. "message""服务异常,请稍后重试" 
  4. 复制代码 

然后,前端根据返回的信息,显示友好的提示页面。

Spring提供了三种方式对异常统一处理:

  1. @ExceptionHandler
  2. 实现HandlerExceptionResolver接口
  3. @ControllerAdvice + @ExceptionHandler

下面我们来实际操作下。

具体实现

@ExceptionHandler

在【统一基类、接口、返回对象设计】这一篇文章中, 定义了Controller的基类BaseController,所以只要在BaseController中使用@ExceptionHandler处理异常, 其它Controller继承BaseController即可。实现如下:

  1. @Slf4j 
  2. public abstract class BaseController { 
  3.  
  4. /** 
  5. * BusinessException 异常处理 
  6. */ 
  7. @ResponseBody 
  8. @ExceptionHandler(BusinessException.class
  9. public ApiResult businessExceptionHandler(BusinessException e) { 
  10. log.error(e.getMessage(), e); 
  11. // do something 
  12. return ApiResult.fail(e.getMessage()); 
  13.  
  14. /** 
  15. * Exception 异常处理 
  16. */ 
  17. @ResponseBody 
  18. @ExceptionHandler(Exception.class
  19. public ApiResult exceptionHandler(Exception e) { 
  20. log.error(e.getMessage(), e); 
  21. return ApiResult.fail("服务异常,请稍后重试"); 
  22. 复制代码 

这里对异常BusinessException和Exception进行了处理, BusinessException是约定的业务异常的基类,若是主动抛出一般都要求是BusinessException的子类,都会被businessExceptionHandler处理。 若是其它异常,可能是意想不到的异常,则会被exceptionHandler处理。

统一处理后,返回结果如下:

从零搭建后端框架——异常统一处理的三种方式

实现HandlerExceptionResolver接口

  1. @Slf4j 
  2. @Component 
  3. public class GlobalHandlerExceptionResolver implements HandlerExceptionResolver { 
  4.  
  5. @Override 
  6. public ModelAndView resolveException(HttpServletRequest request, HttpServletResponse response, Object o, Exception e) { 
  7. log.error(e.getMessage(), e); 
  8. ApiResult apiResult; 
  9. if (e instanceof BusinessException) { 
  10. BusinessException be = (BusinessException) e; 
  11. // do something 
  12. apiResult = ApiResult.fail(be.getMessage()); 
  13. else { 
  14. apiResult = ApiResult.fail("服务异常,请稍后重试"); 
  15. WebUtils.writeJson(response, apiResult); 
  16. return null
  17. 复制代码 

该方式需要实现HandlerExceptionResolver接口,然后将实现类注入到Spring容器中。

但第一种方式中,通过@ResponseBody注解,Spring就帮我们返回了json格式数据,而这需要自己实现。

这里实现了工具类WebUtils,用于返回json数据,如下:

  1. public class WebUtils { 
  2.  
  3. private static final Logger log = LoggerFactory.getLogger(WebUtils.class); 
  4.  
  5. private static Gson gson = new GsonBuilder().serializeNulls().create(); 
  6.  
  7. /** 
  8. * 返回json数据 
  9. * 
  10. * @param response 
  11. * @param object 
  12. */ 
  13. public static void writeJson(HttpServletResponse response, int status, Object object) { 
  14. response.setHeader("Content-Type""application/json;charset=UTF-8"); 
  15. response.setContentType("application/json;charset=UTF-8"); 
  16. response.setStatus(status); 
  17. PrintWriter out = null
  18. try { 
  19. String data = object instanceof String ? (String) object : gson.toJson(object); 
  20. out = response.getWriter(); 
  21. out.print(data); 
  22. out.flush(); 
  23. catch (Exception e) { 
  24. log.error(e.getMessage(), e); 
  25. finally { 
  26. if (out != null) { 
  27. out.close(); 
  28.  
  29. /** 
  30. * 返回json数据 
  31. * 
  32. * @param response 
  33. * @param object 
  34. */ 
  35. public static void writeJson(HttpServletResponse response, Object object) { 
  36. writeJson(response, HttpServletResponse.SC_OK, object); 
  37.  
  38. /** 
  39. * 返回json数据 
  40. * 
  41. * @param response 
  42. * @param object 
  43. */ 
  44. public static void writeJson(ServletResponse response, Object object) { 
  45. if (response instanceof HttpServletResponse) { 
  46. writeJson((HttpServletResponse) response, object); 
  47. 复制代码 

工具类中使用了Gson,需要引用:

  1. <dependency> 
  2. <groupId>com.google.code.gson</groupId> 
  3. <artifactId>gson</artifactId> 
  4. </dependency> 
  5. 复制代码 

@ControllerAdvice + @ExceptionHandler

该方式与第一种方式类似,如下:

  1. @Slf4j 
  2. @ControllerAdvice 
  3. public class GlobalExceptionHandler { 
  4.  
  5. /** 
  6. * BusinessException 异常处理 
  7. */ 
  8. @ResponseBody 
  9. @ExceptionHandler(BusinessException.class
  10. public ApiResult businessExceptionHandler(BusinessException e) { 
  11. log.error(e.getMessage(), e); 
  12. // do something 
  13. return ApiResult.fail(e.getMessage()); 
  14.  
  15. /** 
  16. * Exception 异常处理 
  17. */ 
  18. @ResponseBody 
  19. @ExceptionHandler(Exception.class
  20. public ApiResult exceptionHandler(Exception e) { 
  21. log.error(e.getMessage(), e); 
  22. return ApiResult.fail("服务异常,请稍后重试"); 
  23. 复制代码 

总结

三种方式都能很好对异常进行统一处理,但是一般推荐使用@ControllerAdvice + @ExceptionHandler方式, 这样能够使异常处理与业务逻辑分离,并且不用自己处理Json数据返回。

源码

github.com/zhuqianchan…

 

责任编辑:张燕妮 来源: 今日头条
相关推荐

2024-08-09 08:25:32

Spring流程注解

2023-12-05 14:10:00

接口可读性

2019-08-22 14:02:00

Spring BootRestful APIJava

2020-11-01 17:10:46

异步事件开发前端

2018-04-27 14:18:01

2024-12-03 10:38:15

MapStruct继承关系Java

2020-11-10 10:21:31

架构try...catch代码

2022-05-30 08:03:06

后端参数校验异常处理

2010-02-04 10:33:40

C++异常传递

2012-07-17 09:16:16

SpringSSH

2019-10-09 13:39:39

Python编程语言异常错误

2020-01-02 22:32:00

物联网智能零售RFID

2017-07-31 15:47:50

Zuul统一处理

2021-06-24 08:52:19

单点登录代码前端

2014-12-31 17:42:47

LBSAndroid地图

2019-11-20 18:52:24

物联网智能照明智能恒温器

2021-11-05 21:33:28

Redis数据高并发

2009-08-06 15:26:18

C#异常类型

2010-03-12 17:52:35

Python输入方式

2024-09-04 15:56:28

点赞
收藏

51CTO技术栈公众号