微信公眾號(hào):一個(gè)優(yōu)秀的廢人
如有問題或建議,請(qǐng)后臺(tái)留言殉农,我會(huì)盡力解決你的問題肛炮。
前言
如題惊橱,今天介紹 SpringBoot 是如何統(tǒng)一處理全局異常的蚪腐。SpringBoot 中的全局異常處理主要起作用的兩個(gè)注解是 @ControllerAdvice 和 @ExceptionHandler ,其中 @ControllerAdvice 是組件注解李皇,添加了這個(gè)注解的類能夠攔截 Controller 的請(qǐng)求,而 ExceptionHandler 注解可以設(shè)置全局處理控制里的異常類型來攔截要處理的異常宙枷。 比如:@ExceptionHandler(value = NullPointException.class) 掉房。
準(zhǔn)備工作
- SpringBoot 2.1.3
- IDEA
- JDK 8
依賴配置
<dependencies>
<!-- JPA 依賴 -->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-jpa</artifactId>
</dependency>
<!-- web 依賴 -->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<!-- mysql 連接類 -->
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<scope>runtime</scope>
</dependency>
<!-- lombok 依賴 -->
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
<optional>true</optional>
</dependency>
<!-- 單元測(cè)試依賴 -->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<scope>test</scope>
</dependency>
</dependencies>
配置文件
spring:
# 數(shù)據(jù)庫(kù)相關(guān)
datasource:
driver-class-name: com.mysql.jdbc.Driver
url: jdbc:mysql://127.0.0.1:3306/test?useUnicode=true&characterEncoding=utf8&serverTimezone=UTC&useSSL=true
username: root
password: 123456
jpa:
hibernate:
ddl-auto: update #ddl-auto:設(shè)為 create 表示每次都重新建表
show-sql: true
返回的消息類
public class Message<T> implements Serializable {
/**
* 狀態(tài)碼
*/
private Integer code;
/**
* 返回信息
*/
private String message;
/**
* 返回的數(shù)據(jù)類
*/
private T data;
/**
* 時(shí)間
*/
private Long time;
// getter、setter 以及 構(gòu)造方法略慰丛。卓囚。。
}
工具類
用于處理返回的數(shù)據(jù)以及信息類诅病,代碼注釋很詳細(xì)不說了哪亿。
public class MessageUtil {
/**
* 成功并返回?cái)?shù)據(jù)實(shí)體類
* @param o
* @param <E>
* @return
*/
public static <E>Message<E> ok(E o){
return new Message<>(200, "success", o, new Date().getTime());
}
/**
* 成功粥烁,但無(wú)數(shù)據(jù)實(shí)體類返回
* @return
*/
public static <E>Message<E> ok(){
return new Message<>(200, "success", null, new Date().getTime());
}
/**
* 失敗,有自定義異常返回
* @param code
* @param msg
* @return
*/
public static <E>Message<E> error(Integer code,String msg){
return new Message<>(code, msg, null, new Date().getTime());
}
}
自定義異常
通過繼承 RuntimeException 蝇棉,聲明 code 用于定義不同類型的自定義異常讨阻。主要是用于異常攔截出獲取 code 并將 code 設(shè)置到消息類中返回。
public class CustomException extends RuntimeException{
/**
* 狀態(tài)碼
*/
private Integer code;
public Integer getCode() {
return code;
}
public void setCode(Integer code) {
this.code = code;
}
public CustomException(Integer code, String message){
super(message);
this.code = code;
}
}
異常攔截類
通過加入 @RestControllerAdvice 來聲明該類可攔截 Controller 請(qǐng)求篡殷,同時(shí)在 handle方法加入 @ExceptionHandler 并在該注解中指定要攔截的異常類钝吮。
@RestControllerAdvice // 控制器增強(qiáng)處理(返回 JSON 格式數(shù)據(jù)),添加了這個(gè)注解的類能被 classpath 掃描自動(dòng)發(fā)現(xiàn)
public class ExceptionHandle {
@ExceptionHandler(value = Exception.class) // 捕獲 Controller 中拋出的指定類型的異常板辽,也可以指定其他異常
public <E>Message<E> handler(Exception exception){
if (exception instanceof CustomException){
CustomException customException = (CustomException) exception;
return MessageUtil.error(customException.getCode(), customException.getMessage());
} else {
return MessageUtil.error(120, "異常信息:" + exception.getMessage());
}
}
}
這里只對(duì)自定義異常以及未知異常進(jìn)行處理奇瘦,如果你在某方法中明確知道可能會(huì)拋出某個(gè)異常,可以加多一個(gè)特定的處理劲弦。比如說你明確知道該方法可能拋出 NullPointException 可以追加 NullPointException 的處理:
if (exception instanceof CustomException){
CustomException customException = (CustomException) exception;
return MessageUtil.error(customException.getCode(), customException.getMessage());
} else if (exception instanceof NullPointException ){
return MessageUtil.error(500, "空指針異常信耳标!");
} else {
return MessageUtil.error(120, "異常信息:" + exception.getMessage());
}
controller 層
@RestController
@RequestMapping("/student")
public class StudentController {
@Autowired
private StudentService studentService;
@GetMapping("/{id}")
public Message<Student> findStudentById(@PathVariable("id") Integer id){
if (id < 0){
//測(cè)試自定義錯(cuò)誤
throw new CustomException(110, "參數(shù)不能是負(fù)數(shù)!");
} else if (id == 0){
//硬編碼邑跪,為了測(cè)試
Integer i = 1/id;
return null;
} else {
Student student = studentService.findStudentById(id);
return MessageUtil.ok(student);
}
}
}
完整代碼
https://github.com/turoDog/Demo/tree/master/springboot_exception_demo
如果覺得對(duì)你有幫助次坡,請(qǐng)給個(gè) Star 再走唄,非常感謝呀袱。
Postman 測(cè)試
訪問 http://localhost:8080/student/5 測(cè)試正常返回?cái)?shù)據(jù)結(jié)果贸毕。
訪問 http://localhost:8080/student/0 測(cè)試未知異常的結(jié)果。
訪問 http://localhost:8080/student/-11 測(cè)試自定義異常的結(jié)果夜赵。
后語(yǔ)
如果本文對(duì)你哪怕有一丁點(diǎn)幫助明棍,請(qǐng)幫忙點(diǎn)好看。你的好看是我堅(jiān)持寫作的動(dòng)力寇僧。
另外摊腋,關(guān)注之后在發(fā)送 1024 可領(lǐng)取免費(fèi)學(xué)習(xí)資料。資料內(nèi)容詳情請(qǐng)看這篇舊文:Python嘁傀、C++兴蒸、Java、Linux细办、Go橙凳、前端、算法資料分享