服务器之家:专注于服务器技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - Java教程 - SpringBoot项目优雅的全局异常处理方式(全网最新)

SpringBoot项目优雅的全局异常处理方式(全网最新)

2021-09-12 01:21DT~ Java教程

这篇文章主要介绍了SpringBoot项目优雅的全局异常处理方式(全网最新),文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

前言

在日常项目开发中,异常是常见的,但是如何更高效的处理好异常信息,让我们能快速定位到BUG,是很重要的,不仅能够提高我们的开发效率,还能让你代码看上去更舒服,SpringBoot的项目已经对有一定的异常处理了,但是对于我们开发者而言可能就不太合适了,因此我们需要对这些异常进行统一的捕获并处理。

一、全局异常处理方式一

SpringBoot中,@ControllerAdvice 即可开启全局异常处理,使用该注解表示开启了全局异常的捕获,我们只需在自定义一个方法使用@ExceptionHandler注解然后定义捕获异常的类型即可对这些捕获的异常进行统一的处理。

1.1 自定义全局异常类

  1. /**
  2. * @description: 自定义异常处理
  3. * @author: DT
  4. * @date: 2021/4/19 21:17
  5. * @version: v1.0
  6. */
  7. @ControllerAdvice
  8. public class MyExceptionHandler {
  9.  
  10. @ExceptionHandler(value =Exception.class)
  11. @ResponseBody
  12. public String exceptionHandler(Exception e){
  13. System.out.println("全局异常捕获>>>:"+e);
  14. return "全局异常捕获,错误原因>>>"+e.getMessage();
  15. }
  16. }

1.2 手动抛出异常

  1. @GetMapping("/getById/{userId}")
  2. public CommonResult<User> getById(@PathVariable Integer userId){
  3. // 手动抛出异常
  4. int a = 10/0;
  5. return CommonResult.success(userService.getById(userId));
  6. }

1.3 测试打印

SpringBoot项目优雅的全局异常处理方式(全网最新)

SpringBoot项目优雅的全局异常处理方式(全网最新)

很显然这样的用户体验效果是极差的,虽然这种能够让我们知道异常的原因,但是在很多的情况下来说,可能还是不够人性化,不符合我们的要求。

二、全局异常处理方式二

2.1 定义基础接口类

  1. /**
  2. * @description: 服务接口类
  3. * @author: DT
  4. * @date: 2021/4/19 21:39
  5. */
  6. public interface BaseErrorInfoInterface {
  7.  
  8. /**
  9. * 错误码
  10. * @return
  11. */
  12. String getResultCode();
  13.  
  14. /**
  15. * 错误描述
  16. * @return
  17. */
  18. String getResultMsg();
  19. }

2.2 定义枚举类

  1. /**
  2. * @description: 异常处理枚举类
  3. * @author: DT
  4. * @date: 2021/4/19 21:41
  5. * @version: v1.0
  6. */
  7. public enum ExceptionEnum implements BaseErrorInfoInterface{
  8.  
  9. // 数据操作错误定义
  10. SUCCESS("2000", "成功!"),
  11. BODY_NOT_MATCH("4000","请求的数据格式不符!"),
  12. SIGNATURE_NOT_MATCH("4001","请求的数字签名不匹配!"),
  13. NOT_FOUND("4004", "未找到该资源!"),
  14. INTERNAL_SERVER_ERROR("5000", "服务器内部错误!"),
  15. SERVER_BUSY("5003","服务器正忙,请稍后再试!");
  16.  
  17. /**
  18. * 错误码
  19. */
  20. private final String resultCode;
  21.  
  22. /**
  23. * 错误描述
  24. */
  25. private final String resultMsg;
  26.  
  27. ExceptionEnum(String resultCode, String resultMsg) {
  28. this.resultCode = resultCode;
  29. this.resultMsg = resultMsg;
  30. }
  31.  
  32. @Override
  33. public String getResultCode() {
  34. return resultCode;
  35. }
  36.  
  37. @Override
  38. public String getResultMsg() {
  39. return resultMsg;
  40. }
  41. }

2.3 自定义异常类

  1. /**
  2. * @description: 自定义异常类
  3. * @author: DT
  4. * @date: 2021/4/19 21:44
  5. * @version: v1.0
  6. */
  7. public class BizException extends RuntimeException{
  8.  
  9. private static final long serialVersionUID = 1L;
  10.  
  11. /**
  12. * 错误码
  13. */
  14. protected String errorCode;
  15. /**
  16. * 错误信息
  17. */
  18. protected String errorMsg;
  19.  
  20. public BizException() {
  21. super();
  22. }
  23.  
  24. public BizException(BaseErrorInfoInterface errorInfoInterface) {
  25. super(errorInfoInterface.getResultCode());
  26. this.errorCode = errorInfoInterface.getResultCode();
  27. this.errorMsg = errorInfoInterface.getResultMsg();
  28. }
  29.  
  30. public BizException(BaseErrorInfoInterface errorInfoInterface, Throwable cause) {
  31. super(errorInfoInterface.getResultCode(), cause);
  32. this.errorCode = errorInfoInterface.getResultCode();
  33. this.errorMsg = errorInfoInterface.getResultMsg();
  34. }
  35.  
  36. public BizException(String errorMsg) {
  37. super(errorMsg);
  38. this.errorMsg = errorMsg;
  39. }
  40.  
  41. public BizException(String errorCode, String errorMsg) {
  42. super(errorCode);
  43. this.errorCode = errorCode;
  44. this.errorMsg = errorMsg;
  45. }
  46.  
  47. public BizException(String errorCode, String errorMsg, Throwable cause) {
  48. super(errorCode, cause);
  49. this.errorCode = errorCode;
  50. this.errorMsg = errorMsg;
  51. }
  52.  
  53. public String getErrorCode() {
  54. return errorCode;
  55. }
  56.  
  57. public void setErrorCode(String errorCode) {
  58. this.errorCode = errorCode;
  59. }
  60.  
  61. public String getErrorMsg() {
  62. return errorMsg;
  63. }
  64.  
  65. public void setErrorMsg(String errorMsg) {
  66. this.errorMsg = errorMsg;
  67. }
  68.  
  69. @Override
  70. public Throwable fillInStackTrace() {
  71. return this;
  72. }
  73. }

2.4 自定义数据传输

  1. /**
  2. * @description: 自定义数据传输
  3. * @author: DT
  4. * @date: 2021/4/19 21:47
  5. * @version: v1.0
  6. */
  7. public class ResultResponse {
  8. /**
  9. * 响应代码
  10. */
  11. private String code;
  12.  
  13. /**
  14. * 响应消息
  15. */
  16. private String message;
  17.  
  18. /**
  19. * 响应结果
  20. */
  21. private Object result;
  22.  
  23. public ResultResponse() {
  24. }
  25.  
  26. public ResultResponse(BaseErrorInfoInterface errorInfo) {
  27. this.code = errorInfo.getResultCode();
  28. this.message = errorInfo.getResultMsg();
  29. }
  30.  
  31. public String getCode() {
  32. return code;
  33. }
  34.  
  35. public void setCode(String code) {
  36. this.code = code;
  37. }
  38.  
  39. public String getMessage() {
  40. return message;
  41. }
  42.  
  43. public void setMessage(String message) {
  44. this.message = message;
  45. }
  46.  
  47. public Object getResult() {
  48. return result;
  49. }
  50.  
  51. public void setResult(Object result) {
  52. this.result = result;
  53. }
  54.  
  55. /**
  56. * 成功
  57. *
  58. * @return
  59. */
  60. public static ResultResponse success() {
  61. return success(null);
  62. }
  63.  
  64. /**
  65. * 成功
  66. * @param data
  67. * @return
  68. */
  69. public static ResultResponse success(Object data) {
  70. ResultResponse rb = new ResultResponse();
  71. rb.setCode(ExceptionEnum.SUCCESS.getResultCode());
  72. rb.setMessage(ExceptionEnum.SUCCESS.getResultMsg());
  73. rb.setResult(data);
  74. return rb;
  75. }
  76.  
  77. /**
  78. * 失败
  79. */
  80. public static ResultResponse error(BaseErrorInfoInterface errorInfo) {
  81. ResultResponse rb = new ResultResponse();
  82. rb.setCode(errorInfo.getResultCode());
  83. rb.setMessage(errorInfo.getResultMsg());
  84. rb.setResult(null);
  85. return rb;
  86. }
  87.  
  88. /**
  89. * 失败
  90. */
  91. public static ResultResponse error(String code, String message) {
  92. ResultResponse rb = new ResultResponse();
  93. rb.setCode(code);
  94. rb.setMessage(message);
  95. rb.setResult(null);
  96. return rb;
  97. }
  98.  
  99. /**
  100. * 失败
  101. */
  102. public static ResultResponse error( String message) {
  103. ResultResponse rb = new ResultResponse();
  104. rb.setCode("-1");
  105. rb.setMessage(message);
  106. rb.setResult(null);
  107. return rb;
  108. }
  109.  
  110. @Override
  111. public String toString() {
  112. return JSONObject.toJSONString(this);
  113. }
  114.  
  115. }

2.5 自定义全局异常处理

  1. /**
  2. * @description: 自定义异常处理
  3. * @author: DT
  4. * @date: 2021/4/19 21:51
  5. * @version: v1.0
  6. */
  7. @ControllerAdvice
  8. public class GlobalExceptionHandler {
  9.  
  10. private static final Logger logger = LoggerFactory.getLogger(GlobalExceptionHandler.class);
  11.  
  12. /**
  13. * 处理自定义的业务异常
  14. * @param req
  15. * @param e
  16. * @return
  17. */
  18. @ExceptionHandler(value = BizException.class)
  19. @ResponseBody
  20. public ResultResponse bizExceptionHandler(HttpServletRequest req, BizException e){
  21. logger.error("发生业务异常!原因是:{}",e.getErrorMsg());
  22. return ResultResponse.error(e.getErrorCode(),e.getErrorMsg());
  23. }
  24.  
  25. /**
  26. * 处理空指针的异常
  27. * @param req
  28. * @param e
  29. * @return
  30. */
  31. @ExceptionHandler(value =NullPointerException.class)
  32. @ResponseBody
  33. public ResultResponse exceptionHandler(HttpServletRequest req, NullPointerException e){
  34. logger.error("发生空指针异常!原因是:",e);
  35. return ResultResponse.error(ExceptionEnum.BODY_NOT_MATCH);
  36. }
  37.  
  38. /**
  39. * 处理其他异常
  40. * @param req
  41. * @param e
  42. * @return
  43. */
  44. @ExceptionHandler(value =Exception.class)
  45. @ResponseBody
  46. public ResultResponse exceptionHandler(HttpServletRequest req, Exception e){
  47. logger.error("未知异常!原因是:",e);
  48. return ResultResponse.error(ExceptionEnum.INTERNAL_SERVER_ERROR);
  49. }
  50. }

2.6 测试代码

  1. @PostMapping("/add")
  2. public boolean add(@RequestBody User user) {
  3. //如果姓名为空就手动抛出一个自定义的异常!
  4. if(user.getName()==null){
  5. throw new BizException("-1","用户姓名不能为空!");
  6. }
  7. return true;
  8. }

SpringBoot项目优雅的全局异常处理方式(全网最新)

  1. @PutMapping("/update")
  2. public boolean update(@RequestBody User user) {
  3. //这里故意造成一个空指针的异常,并且不进行处理
  4. String str = null;
  5. str.equals("111");
  6. return true;
  7. }

SpringBoot项目优雅的全局异常处理方式(全网最新)

  1. @DeleteMapping("/delete")
  2. public boolean delete(@RequestBody User user) {
  3. //这里故意造成一个异常,并且不进行处理
  4. Integer.parseInt("abc123");
  5. return true;
  6. }

SpringBoot项目优雅的全局异常处理方式(全网最新)

如果我们想捕获这个类型转换异常,是不是再添加一个遗产处理方法就可了。

SpringBoot项目优雅的全局异常处理方式(全网最新)

  1. /**
  2. * 处理类型转换异常
  3. * @param req
  4. * @param e
  5. * @return
  6. */
  7. @ExceptionHandler(value = NumberFormatException.class)
  8. @ResponseBody
  9. public ResultResponse exceptionHandler(HttpServletRequest req, NumberFormatException e){
  10. logger.error("发生类型转换异常!原因是:",e);
  11. return ResultResponse.error(ExceptionEnum.PARAMS_NOT_CONVERT);
  12. }
  1. PARAMS_NOT_CONVERT("4002","类型转换不对!"),

SpringBoot项目优雅的全局异常处理方式(全网最新)

自定义全局异常处理除了可以处理上述的数据格式之外,也可以处理页面的跳转,只需在新增的异常方法的返回处理上填写该跳转的路径并不使用ResponseBody 注解即可。

总结

异常处理,能够减少代码的重复度和复杂度,有利于代码的维护,并且能够快速定位到BUG,大大提高我们的开发效率。

到此这篇关于SpringBoot项目优雅的全局异常处理方式(全网最新)的文章就介绍到这了,更多相关SpringBoot 全局异常处理 内容请搜索服务器之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持服务器之家!

原文链接:https://blog.csdn.net/qq_41107231/article/details/115874974

延伸 · 阅读

精彩推荐
  • Java教程Java8中Stream使用的一个注意事项

    Java8中Stream使用的一个注意事项

    最近在工作中发现了对于集合操作转换的神器,java8新特性 stream,但在使用中遇到了一个非常重要的注意点,所以这篇文章主要给大家介绍了关于Java8中S...

    阿杜7472021-02-04
  • Java教程小米推送Java代码

    小米推送Java代码

    今天小编就为大家分享一篇关于小米推送Java代码,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧...

    富贵稳中求8032021-07-12
  • Java教程Java实现抢红包功能

    Java实现抢红包功能

    这篇文章主要为大家详细介绍了Java实现抢红包功能,采用多线程模拟多人同时抢红包,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙...

    littleschemer13532021-05-16
  • Java教程xml与Java对象的转换详解

    xml与Java对象的转换详解

    这篇文章主要介绍了xml与Java对象的转换详解的相关资料,需要的朋友可以参考下...

    Java教程网2942020-09-17
  • Java教程Java BufferWriter写文件写不进去或缺失数据的解决

    Java BufferWriter写文件写不进去或缺失数据的解决

    这篇文章主要介绍了Java BufferWriter写文件写不进去或缺失数据的解决方案,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望...

    spcoder14552021-10-18
  • Java教程升级IDEA后Lombok不能使用的解决方法

    升级IDEA后Lombok不能使用的解决方法

    最近看到提示IDEA提示升级,寻思已经有好久没有升过级了。升级完毕重启之后,突然发现好多错误,本文就来介绍一下如何解决,感兴趣的可以了解一下...

    程序猿DD9332021-10-08
  • Java教程20个非常实用的Java程序代码片段

    20个非常实用的Java程序代码片段

    这篇文章主要为大家分享了20个非常实用的Java程序片段,对java开发项目有所帮助,感兴趣的小伙伴们可以参考一下 ...

    lijiao5352020-04-06
  • Java教程Java使用SAX解析xml的示例

    Java使用SAX解析xml的示例

    这篇文章主要介绍了Java使用SAX解析xml的示例,帮助大家更好的理解和学习使用Java,感兴趣的朋友可以了解下...

    大行者10067412021-08-30