经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » Java相关 » Spring Boot » 查看文章
Spring Boot 统一异常这样处理和剖析,安否?
来源:cnblogs  作者:tan日拱一兵  时间:2019/8/20 9:02:13  对本文有异议

话说异常

「欲渡黄河冰塞川,将登太行雪满天」,无论生活还是计算机世界难免发生异常,上一篇文章RESTful API 返回统一JSON数据格式 说明了统一返回的处理,这是请求一切正常的情形;这篇文章将说明如何统一处理异常,以及其背后的实现原理,老套路,先实现,后说明原理,有了上一篇文章的铺底,相信,理解这篇文章就驾轻就熟了

实现

新建业务异常

新建 BusinessException.class 类表示业务异常,注意这是一个 Runtime 异常

  1. @Data
  2. @AllArgsConstructor
  3. public final class BusinessException extends RuntimeException {
  4. private String errorCode;
  5. private String errorMsg;
  6. }

添加统一异常处理静态方法

在 CommonResult 类中添加静态方法 errorResult 用于接收异常码和异常消息:

  1. public static <T> CommonResult<T> errorResult(String errorCode, String errorMsg){
  2. CommonResult<T> commonResult = new CommonResult<>();
  3. commonResult.errorCode = errorCode;
  4. commonResult.errorMsg = errorMsg;
  5. commonResult.status = -1;
  6. return commonResult;
  7. }

配置

同样要用到 @RestControllerAdvice 注解,将统一异常添加到配置中:

  1. @RestControllerAdvice("com.example.unifiedreturn.api")
  2. static class UnifiedExceptionHandler{
  3. @ExceptionHandler(BusinessException.class)
  4. public CommonResult<Void> handleBusinessException(BusinessException be){
  5. return CommonResult.errorResult(be.getErrorCode(), be.getErrorMsg());
  6. }
  7. }

三部搞定,到这里无论是 Controller 还是 Service 中,只要抛出 BusinessException, 我们都会返回给前端一个统一数据格式

测试

将 UserController 中的方法进行改造,直接抛出异常:

  1. @GetMapping("/{id}")
  2. public UserVo getUserById(@PathVariable Long id){
  3. throw new BusinessException("1001", "根据ID查询用户异常");
  4. }

浏览器中输入: http://localhost:8080/users/1

在 Service 中抛出异常:

  1. @Service
  2. public class UserServiceImpl implements UserService {
  3. /**
  4. * 根据用户ID查询用户
  5. *
  6. * @param id
  7. * @return
  8. */
  9. @Override
  10. public UserVo getUserById(Long id) {
  11. throw new BusinessException("1001", "根据ID查询用户异常");
  12. }
  13. }

运行是得到同样的结果,所以我们尽可能的抛出异常吧 (作为一个程序猿这种心理很可拍)

解剖实现过程

解剖这个过程是相当纠结的,为了更好的说(yin)明(wei)问(wo)题(lan),我要说重中之重了,真心希望看该文章的童鞋自己去案发现场发现线索
还是在 WebMvcConfigurationSupport 类中实例化了 HandlerExceptionResolver Bean

  1. @Bean
  2. public HandlerExceptionResolver handlerExceptionResolver() {
  3. List<HandlerExceptionResolver> exceptionResolvers = new ArrayList<>();
  4. configureHandlerExceptionResolvers(exceptionResolvers);
  5. if (exceptionResolvers.isEmpty()) {
  6. addDefaultHandlerExceptionResolvers(exceptionResolvers);
  7. }
  8. extendHandlerExceptionResolvers(exceptionResolvers);
  9. HandlerExceptionResolverComposite composite = new HandlerExceptionResolverComposite();
  10. composite.setOrder(0);
  11. composite.setExceptionResolvers(exceptionResolvers);
  12. return composite;
  13. }

和上一篇文章一毛一样的套路,ExceptionHandlerExceptionResolver 实现了 InitializingBean 接口,重写了 afterPropertiesSet 方法:

  1. @Override
  2. public void afterPropertiesSet() {
  3. // Do this first, it may add ResponseBodyAdvice beans
  4. initExceptionHandlerAdviceCache();
  5. ...
  6. }
  7. private void initExceptionHandlerAdviceCache() {
  8. if (getApplicationContext() == null) {
  9. return;
  10. }
  11. List<ControllerAdviceBean> adviceBeans = ControllerAdviceBean.findAnnotatedBeans(getApplicationContext());
  12. AnnotationAwareOrderComparator.sort(adviceBeans);
  13. for (ControllerAdviceBean adviceBean : adviceBeans) {
  14. Class<?> beanType = adviceBean.getBeanType();
  15. if (beanType == null) {
  16. throw new IllegalStateException("Unresolvable type for ControllerAdviceBean: " + adviceBean);
  17. }
  18. // 重点看这个构造方法
  19. ExceptionHandlerMethodResolver resolver = new ExceptionHandlerMethodResolver(beanType);
  20. if (resolver.hasExceptionMappings()) {
  21. this.exceptionHandlerAdviceCache.put(adviceBean, resolver);
  22. }
  23. if (ResponseBodyAdvice.class.isAssignableFrom(beanType)) {
  24. this.responseBodyAdvice.add(adviceBean);
  25. }
  26. }
  27. }

重点看上面我用注释标记的构造方法,代码很好懂,仔细看看吧,其实就是筛选出我们用 @ExceptionHandler 注解标记的方法并放到集合当中,用于后续全局异常捕获的匹配

  1. /**
  2. * A constructor that finds {@link ExceptionHandler} methods in the given type.
  3. * @param handlerType the type to introspect
  4. */
  5. public ExceptionHandlerMethodResolver(Class<?> handlerType) {
  6. for (Method method : MethodIntrospector.selectMethods(handlerType, EXCEPTION_HANDLER_METHODS)) {
  7. for (Class<? extends Throwable> exceptionType : detectExceptionMappings(method)) {
  8. addExceptionMapping(exceptionType, method);
  9. }
  10. }
  11. }
  12. /**
  13. * Extract exception mappings from the {@code @ExceptionHandler} annotation first,
  14. * and then as a fallback from the method signature itself.
  15. */
  16. @SuppressWarnings("unchecked")
  17. private List<Class<? extends Throwable>> detectExceptionMappings(Method method) {
  18. List<Class<? extends Throwable>> result = new ArrayList<>();
  19. detectAnnotationExceptionMappings(method, result);
  20. if (result.isEmpty()) {
  21. for (Class<?> paramType : method.getParameterTypes()) {
  22. if (Throwable.class.isAssignableFrom(paramType)) {
  23. result.add((Class<? extends Throwable>) paramType);
  24. }
  25. }
  26. }
  27. if (result.isEmpty()) {
  28. throw new IllegalStateException("No exception types mapped to " + method);
  29. }
  30. return result;
  31. }
  32. private void detectAnnotationExceptionMappings(Method method, List<Class<? extends Throwable>> result) {
  33. ExceptionHandler ann = AnnotatedElementUtils.findMergedAnnotation(method, ExceptionHandler.class);
  34. Assert.state(ann != null, "No ExceptionHandler annotation");
  35. result.addAll(Arrays.asList(ann.value()));
  36. }

到这里,我们用 @RestControllerAdvice@ExceptionHandler 注解就会被 Spring 扫描到上下文,供我们使用

让我们回到你最熟悉的调用的入口 DispatcherServlet 类的 doDispatch 方法:

  1. protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throws Exception {
  2. ...
  3. try {
  4. ModelAndView mv = null;
  5. Exception dispatchException = null;
  6. try {
  7. ...
  8. // 当请求发生异常,该方法会通过 catch 捕获异常
  9. mv = ha.handle(processedRequest, response, mappedHandler.getHandler());
  10. ...
  11. }
  12. catch (Exception ex) {
  13. dispatchException = ex;
  14. }
  15. catch (Throwable err) {
  16. // As of 4.3, we're processing Errors thrown from handler methods as well,
  17. // making them available for @ExceptionHandler methods and other scenarios.
  18. dispatchException = new NestedServletException("Handler dispatch failed", err);
  19. }
  20. // 调用该方法分析捕获的异常
  21. processDispatchResult(processedRequest, response, mappedHandler, mv, dispatchException);
  22. }
  23. ...
  24. }

接下来,我们来看 processDispatchResult 方法,这里只要展示调用栈你就会眼前一亮了,又是为了返回统一格式数据:

总结

上一篇文章的返回统一数据格式是基础,当异常情况发生时,只不过需要将异常信息提取出来。本文主要为了说明问题,剖析原理,好多地方设计方式是不可取,比如我们最好将异常封装在一个 Enum 类,通过 enum 对象抛出异常等,如果你用到这些,去完善你的设计方案吧

回复 「demo」,打开链接,查看文件夹 「unifiedreturn」下内容,获取完整代码,更好阅读体验:
https://fraseryu.github.io/2019/08/09/ru-he-tong-yi-chu-li-yi-chang-bing-fan-hui-tong-yi-ge-shi/

附加说明

之前看到的一本书对异常的分类让我印象深刻,在此摘录一小段分享给大家:

结合出国旅行的例子说明异常分类:

  • 机场地震,属于不可抗力,对应异常分类中的 Error,在制订出行计划时,根本不需要把这个部分的异常考虑进去
  • 堵车属于 checked 异常,应对这种异常,我们可以提前出发,或者改签机票。而飞机延误异常,虽然也需要 check,但我们无能为力,只能持续关注航班动态
  • 没有带护照,明显属于可提前预测的异常,只要出发前检查即可避免;去机场路上车子抛锚,这个异常是突发的,虽然难以预料,但是必须处理,属于需要捕捉的异常,可以通过更换交通工具;应对检票机器故障属于 可透出异常,交由航空公司处理,我们无须关心

灵魂追问

  1. 这两篇文章,你学到了哪些设计模式?
  2. 你能熟练的使用反射吗?当看源码是会看到很多反射的应用
  3. 你了解 Spring CGLIB 吗?它的工作原理是什么?

提高效率工具

JSON-Viewer

JSON-Viewer 是 Chrome 浏览器的插件,用于快速解析及格式化 json 内容,在 Chrome omnibox(多功能输入框)输入json-viewer + TAB ,将 json 内容拷贝进去,然后输入回车键,将看到结构清晰的 json 数据,同时可以自定义主题

另外,前端人员打开开发者工具,双击请求链接,会自动将 response 中的 json 数据解析出来,非常方便

原文链接:http://www.cnblogs.com/FraserYu/p/11380878.html

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号