SpringBoot的全局异常处理的优雅吃法!要进来学习下吗

SpringBoot全局异常准备
开发准备
环境要求 JDK :1.8 SpringBoot :1.5.17.RELEASE

首先还是Maven的相关依赖:

<properties><project.build.sourceEncoding>UTF-8</project.build.sourceEncoding><java.version>1.8</java.version><maven.compiler.source>1.8</maven.compiler.source><maven.compiler.target>1.8</maven.compiler.target></properties><parent><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-parent</artifactId><version>1.5.17.RELEASE</version><relativePath /></parent><dependencies><!-- Spring Boot Web 依赖 核心 --><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId></dependency><!-- Spring Boot Test 依赖 --><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-test</artifactId><scope>test</scope></dependency><dependency><groupId>com.alibaba</groupId><artifactId>fastjson</artifactId><version>1.2.41</version></dependency></dependencies>

配置文件这块基本不需要更改,全局异常的处理只需在代码中实现即可。

代码编写
SpringBoot的项目已经对有一定的异常处理了,但是对于我们开发者而言可能就不太合适了,因此我们需要对这些异常进行统一的捕获并处理。SpringBoot中有一个 ControllerAdvice 的注解,使用该注解表示开启了全局异常的捕获,我们只需在自定义一个方法使用 ExceptionHandler 注解然后定义捕获异常的类型即可对这些捕获的异常进行统一的处理。

我们根据下面的这个示例来看该注解是如何使用吧。

示例代码:

@ControllerAdvice
public class MyExceptionHandler {@ExceptionHandler(value =Exception.class)public String exceptionHandler(Exception e){System.out.println("未知异常!原因是:"+e);return e.getMessage();}
}

上述的示例中,我们对捕获的异常进行简单的二次处理,返回异常的信息,虽然这种能够让我们知道异常的原因,但是在很多的情况下来说,可能还是不够人性化,不符合我们的要求。那么我们这里可以通过自定义的异常类以及枚举类来实现我们想要的那种数据吧。

自定义基础接口类

首先定义一个基础的接口类,自定义的错误描述枚举类需实现该接口。 代码如下:

public interface BaseErrorInfoInterface {/** 错误码*/String getResultCode();/** 错误描述*/String getResultMsg();
}

自定义枚举类

然后我们这里在自定义一个枚举类,并实现该接口。 代码如下:

public enum CommonEnum implements BaseErrorInfoInterface {// 数据操作错误定义SUCCESS("200", "成功!"), BODY_NOT_MATCH("400","请求的数据格式不符!"),SIGNATURE_NOT_MATCH("401","请求的数字签名不匹配!"),NOT_FOUND("404", "未找到该资源!"), INTERNAL_SERVER_ERROR("500", "服务器内部错误!"),SERVER_BUSY("503","服务器正忙,请稍后再试!");/** 错误码 */private String resultCode;/** 错误描述 */private String resultMsg;CommonEnum(String resultCode, String resultMsg) {this.resultCode = resultCode;this.resultMsg = resultMsg;}@Overridepublic String getResultCode() {return resultCode;}@Overridepublic String getResultMsg() {return resultMsg;}}

自定义异常类

然后我们在来自定义一个异常类,用于处理我们发生的业务异常。 代码如下:

public class BizException extends RuntimeException {private static final long serialVersionUID = 1L;/*** 错误码*/protected String errorCode;/*** 错误信息*/protected String errorMsg;public BizException() {super();}public BizException(BaseErrorInfoInterface errorInfoInterface) {super(errorInfoInterface.getResultCode());this.errorCode = errorInfoInterface.getResultCode();this.errorMsg = errorInfoInterface.getResultMsg();}public BizException(BaseErrorInfoInterface errorInfoInterface, Throwable cause) {super(errorInfoInterface.getResultCode(), cause);this.errorCode = errorInfoInterface.getResultCode();this.errorMsg = errorInfoInterface.getResultMsg();}public BizException(String errorMsg) {super(errorMsg);this.errorMsg = errorMsg;}public BizException(String errorCode, String errorMsg) {super(errorCode);this.errorCode = errorCode;this.errorMsg = errorMsg;}public BizException(String errorCode, String errorMsg, Throwable cause) {super(errorCode, cause);this.errorCode = errorCode;this.errorMsg = errorMsg;}public String getErrorCode() {return errorCode;}public void setErrorCode(String errorCode) {this.errorCode = errorCode;}public String getErrorMsg() {return errorMsg;}public void setErrorMsg(String errorMsg) {this.errorMsg = errorMsg;}public String getMessage() {return errorMsg;}@Overridepublic Throwable fillInStackTrace() {return this;}}

自定义数据格式

顺便这里我们定义一下数据的传输格式。 代码如下:

public class ResultBody {/*** 响应代码*/private String code;/*** 响应消息*/private String message;/*** 响应结果*/private Object result;public ResultBody() {}public ResultBody(BaseErrorInfoInterface errorInfo) {this.code = errorInfo.getResultCode();this.message = errorInfo.getResultMsg();}public String getCode() {return code;}public void setCode(String code) {this.code = code;}public String getMessage() {return message;}public void setMessage(String message) {this.message = message;}public Object getResult() {return result;}public void setResult(Object result) {this.result = result;}/*** 成功* * @return*/public static ResultBody success() {return success(null);}/*** 成功* @param data* @return*/public static ResultBody success(Object data) {ResultBody rb = new ResultBody();rb.setCode(CommonEnum.SUCCESS.getResultCode());rb.setMessage(CommonEnum.SUCCESS.getResultMsg());rb.setResult(data);return rb;}/*** 失败*/public static ResultBody error(BaseErrorInfoInterface errorInfo) {ResultBody rb = new ResultBody();rb.setCode(errorInfo.getResultCode());rb.setMessage(errorInfo.getResultMsg());rb.setResult(null);return rb;}/*** 失败*/public static ResultBody error(String code, String message) {ResultBody rb = new ResultBody();rb.setCode(code);rb.setMessage(message);rb.setResult(null);return rb;}/*** 失败*/public static ResultBody error( String message) {ResultBody rb = new ResultBody();rb.setCode("-1");rb.setMessage(message);rb.setResult(null);return rb;}@Overridepublic String toString() {return JSONObject.toJSONString(this);}}

自定义全局异常处理类

最后我们在来编写一个自定义全局异常处理的类。 代码如下:

@ControllerAdvice
public class GlobalExceptionHandler {private static final Logger logger = LoggerFactory.getLogger(GlobalExceptionHandler.class);/*** 处理自定义的业务异常* @param req* @param e* @return*/@ExceptionHandler(value = BizException.class)  @ResponseBody  public  ResultBody bizExceptionHandler(HttpServletRequest req, BizException e){logger.error("发生业务异常!原因是:{}",e.getErrorMsg());return ResultBody.error(e.getErrorCode(),e.getErrorMsg());}/*** 处理空指针的异常* @param req* @param e* @return*/@ExceptionHandler(value =NullPointerException.class)@ResponseBodypublic ResultBody exceptionHandler(HttpServletRequest req, NullPointerException e){logger.error("发生空指针异常!原因是:",e);return ResultBody.error(CommonEnum.BODY_NOT_MATCH);}/*** 处理其他异常* @param req* @param e* @return*/@ExceptionHandler(value =Exception.class)@ResponseBodypublic ResultBody exceptionHandler(HttpServletRequest req, Exception e){logger.error("未知异常!原因是:",e);return ResultBody.error(CommonEnum.INTERNAL_SERVER_ERROR);}
}

因为这里我们只是用于做全局异常处理的功能实现以及测试,所以这里我们只需在添加一个实体类和一个控制层类即可。

实体类

又是万能的用户表 ( ▽ )

代码如下:

public class User implements Serializable{private static final long serialVersionUID = 1L;/** 编号 */private int id;/** 姓名 */private String name;/** 年龄 */private int age;public User(){}public int getId() {return id;}public void setId(int id) {this.id = id;}public String getName() {return name;}public void setName(String name) {this.name = name;}public int getAge() {return age;}public void setAge(int age) {this.age = age;}public String toString() {return JSONObject.toJSONString(this);}
}

Controller 控制层

控制层这边也比较简单,使用Restful风格实现的CRUD功能,不同的是这里我故意弄出了一些异常,好让这些异常被捕获到然后处理。这些异常中,有自定义的异常抛出,也有空指针的异常抛出,当然也有不可预知的异常抛出(这里我用类型转换异常代替),那么我们在完成代码编写之后,看看这些异常是否能够被捕获处理成功吧!

代码如下:

@RestController
@RequestMapping(value = "/api")
public class UserRestController {@PostMapping("/user")public boolean insert(@RequestBody User user) {System.out.println("开始新增...");//如果姓名为空就手动抛出一个自定义的异常!if(user.getName()==null){throw  new BizException("-1","用户姓名不能为空!");}return true;}@PutMapping("/user")public boolean update(@RequestBody User user) {System.out.println("开始更新...");//这里故意造成一个空指针的异常,并且不进行处理String str=null;str.equals("111");return true;}@DeleteMapping("/user")public boolean delete(@RequestBody User user)  {System.out.println("开始删除...");//这里故意造成一个异常,并且不进行处理Integer.parseInt("abc123");return true;}@GetMapping("/user")public List<User> findByUser(User user) {System.out.println("开始查询...");List<User> userList =new ArrayList<>();User user2=new User();user2.setId(1L);user2.setName("xuwujing");user2.setAge(18);userList.add(user2);return userList;}}

App 入口

和普通的SpringBoot项目基本一样。

代码如下:

@SpringBootApplication
public class App
{public static void main( String[] args ){SpringApplication.run(App.class, args);System.out.println("程序正在运行...");}
}

功能测试
我们成功启动该程序之后,使用Postman工具来进行接口测试。

首先进行查询,查看程序正常运行是否ok,使用GET 方式进行请求。

“GET http://localhost:8181/api/user”返回参数为:“{"id":1,"name":"xuwujing","age":18}”

示例图:

可以看到程序正常返回,并没有因自定义的全局异常而影响。

然后我们再来测试下自定义的异常是否能够被正确的捕获并处理。

使用POST方式进行请求

“POST http://localhost:8181/api/user”Body参数为:“{"id":1,"age":18}”返回参数为:“{"code":"-1","message":"用户姓名不能为空!","result":null}”

示例图:

可以看出将我们抛出的异常进行数据封装,然后将异常返回出来。

然后我们再来测试下空指针异常是否能够被正确的捕获并处理。在自定义全局异常中,我们除了定义空指针的异常处理,也定义最高级别之一的Exception异常,那么这里发生了空指针异常之后,它是回优先使用哪一个呢?这里我们来测试下。

使用PUT方式进行请求。

“PUT http://localhost:8181/api/user”Body参数为:“{“id”:1,“age”:18}”返回参数为:“{“code”:“400”,“message”:“请求的数据格式不符!”,“result”:null}”

示例图:

我们可以看到这里的的确是返回空指针的异常护理,可以得出全局异常处理优先处理子类的异常。

那么我们在来试试未指定其异常的处理,看该异常是否能够被捕获。

使用DELETE方式进行请求。

“DELETE http://localhost:8181/api/user”Body参数为:“{“id”:1}”返回参数为:“{“code”:“500”,“message”:“服务器内部错误!”,“result”:null}”

这里可以看到它使用了我们在自定义全局异常处理类中的Exception异常处理的方法。到这里,测试就结束了。顺便再说一下,自义定全局异常处理除了可以处理上述的数据格式之外,也可以处理页面的跳转,只需在新增的异常方法的返回处理上填写该跳转的路径并不使用 ResponseBody 注解即可。

细心的同学也许发现了在 GlobalExceptionHandler 类中使用的是 ControllerAdvice 注解,而非 RestControllerAdvice 注解,如果是用的 RestControllerAdvice 注解,它会将数据自动转换成JSON格式,这种于 Controller 和 RestController 类似,所以我们在使用全局异常处理的之后可以进行灵活的选择处理。

小结
关于SpringBoot优雅的全局异常处理的方法,就到这里啦,有什么不懂的可在评论区交流,如有不妥,欢迎指正!

码字不易 请大家给我点赞+关注+评论!一条龙

文章来源:https://www.jianshu.com/p/dc99ffc6117d

我是Remi酱~~

SpringBoot的全局异常处理的优雅吃法!要进来学习下吗相关推荐

  1. 看看人家 SpringBoot 的全局异常处理多么优雅...

    1 SpringBoot全局异常准备 说明:如果想直接获取工程那么可以直接跳到底部,通过链接下载工程代码. 开发准备 环境要求JDK:1.8SpringBoot:1.5.17.RELEASE 首先还是 ...

  2. SpringBoot集成全局异常处理

    2019独角兽企业重金招聘Python工程师标准>>> SpringBoot集成全局异常处理 前言 对于通常的MVC项目,大量的异常需要我们去处理,如此一来,我们的 Controll ...

  3. 看看别人家 SpringBoot 的全局异常处理,多么优雅....

    今日推荐编程大佬是否能记住代码,不用百度就能啪啪啪敲出来么?在 IDEA 中的各种调试技巧,轻松定位 Bug(超级全面) 后端接口如何提高性能? 16 个写代码的好习惯 为什么不推荐使用BeanUti ...

  4. 看看人家 SpringBoot 的全局异常处理,多么优雅...

    点击关注公众号,Java干货及时送达 本篇文章主要介绍的是SpringBoot项目进行全局异常的处理. SpringBoot全局异常准备 说明:如果想直接获取工程那么可以直接跳到底部,通过链接下载工程 ...

  5. springboot定义全局异常处理类GlobalExceptionHandler

    全局异常处理:GlobalExceptionHandler 定义全局异常处理类 以下为扩展部分 返回参数封装 引用示例 定义全局异常处理类 注意:如果需要返回页面,则要删掉@ResponseBody注 ...

  6. Springboot 自定义全局异常处理

    在项目根包目录下新建 exception.base 包 新建BaseException 继承 RuntimeException package com.ddz.errordemo.handler;/* ...

  7. springboot基于全局异常处理的简单日志打印

    log.error() Logger类下有多个不同的error方法,根据传入参数的个数及类型的不同,自动选择不同的重载方法. 当error(Object obj)只传入一个参数时会将异常对象作为Obj ...

  8. springBoot 全局异常处理 报错 : Could not resolve method parameter at index 0 in public .....

    在学习 springboot 的全局异常处理时   ,我写了一个处理异常类 @ControllerAdvice public class ErrorCatchController {@Exceptio ...

  9. springboot全局异常处理_SpringBoot:如何优雅地处理全局异常

    之前用springboot的时候,只知道捕获异常使用try{}catch,一个接口一个try{}catch,这也是大多数开发人员异常处理的常用方式,虽然屡试不爽,但会造成一个问题,就是一个Contro ...

最新文章

  1. 大年初二,今年过年你选择在男方家过还是女方家过?
  2. 剑指offer 算法 (知识迁移能力)
  3. 字节跳动花50亿买了个什么?
  4. C++设计模式--单例模式(Singleton)及单例通用模板
  5. QT的QPicture类的使用
  6. windows下php7安装redis扩展
  7. java程序中执行maven_java – 将一个enviornment变量传递给Maven中的已执行进程
  8. 物联网有哪些技术 物联网跟云计算人工智能有什么关系
  9. 微软发布了Visual Studio 2022 RC版,并将在11月8日发布正式版
  10. 高斯、柯西、拉格朗日都还在的话,他们应该最喜欢这个公众号
  11. python求阶乘之和_python计算阶乘前n项和
  12. 完全CSS实现鼠标移上出现层的效果(超简单)
  13. Android 自定义View实现画背景和前景(ViewGroup篇)
  14. C++函数模板5分钟入门
  15. spark学习-41-Spark的块传输服务BlockTransferService
  16. linux 对文件类型的判断
  17. 为什么不推荐使用BeanUtils属性转换工具,老程序员都不使用!
  18. 代码阅读工具-global
  19. 微博表情包大全,截止2022年5月
  20. 计算机网络谢希仁第七版课后习题答案

热门文章

  1. OO实现ALV TABLE 六:ALV的弹出窗口形式
  2. 大商超的2020:转型、收紧、试新
  3. 迎来“进化”风口的电竞,能否在未来站上传统体育的肩膀?
  4. InMobi收购美国Sprint旗下数据和广告公司Pinsight Media
  5. java前后端用json传值_前后端——json的传值与接收(springMvc)
  6. 操作系统使用户和计算机的接口 对吗,计算机操作系统教程--核心与设计原理习题10答案...
  7. centos8更换yum源_基于yum进行linux系统升级--从Centos7升级到Centos8
  8. 企业计算机能力,以IT企业需求为导向的计算机人才培养模式研究
  9. c语言农夫过河游戏代码解释,农夫过河C++代码
  10. linux如何关闭 lvm管理,Linux之LVM管理