SpringBoot中@RestControllerAdvice注解實(shí)現(xiàn)全局異常處理類
需求
springboot中使用@RestControllerAdvice注解,完成優(yōu)雅的全局異常處理類,可以針對(duì)所有異常類型先進(jìn)行通用處理后再對(duì)特定異常類型進(jìn)行不同的處理操作。
注解講解
@RestControllerAdvice注解
@RestControllerAdvice是一個(gè)用于定義全局異常處理器的注解。當(dāng)應(yīng)用程序內(nèi)發(fā)生未捕獲的異常時(shí),全局異常處理器將捕獲該異常并返回對(duì)應(yīng)的響應(yīng),以避免應(yīng)用程序崩潰。它可以處理所有控制器中拋出的異常,包括請(qǐng)求處理方法中的異常、控制器構(gòu)造函數(shù)中的異常等。
@RestControllerAdvice注解是@ControllerAdvice和@ResponseBody注解的組合,它的作用是將所有的異常處理結(jié)果都以JSON格式返回給客戶端。
具體來說,當(dāng)控制器中發(fā)生異常時(shí),SpringBoot會(huì)在全局異常處理器中查找與異常匹配的處理方法,并執(zhí)行該方法來處理異常。
處理方法可以返回任何類型的值,如果返回對(duì)象是DataVO類型,則會(huì)將其轉(zhuǎn)換為JSON格式并返回給客戶端。
如果返回值是String類型,則會(huì)將其解釋為視圖名稱,并使用視圖解析器來解析視圖并生成HTML響應(yīng)。
因此,使用@RestControllerAdvice注解可以方便地定義全局異常處理器,并將所有異常處理結(jié)果以JSON格式返回給客戶端。
@ExceptionHandler注解
@ExceptionHandler是一個(gè)注解,用于定義異常處理方法。
當(dāng)控制器中發(fā)生異常時(shí),Spring Boot會(huì)在@ControllerAdvice或@RestControllerAdvice注解的類中查找與異常匹配的@ExceptionHandler注解標(biāo)記的方法,并執(zhí)行該方法來處理異常。
@ExceptionHandler注解可以定義一個(gè)或多個(gè)異常類型,并將它們映射到對(duì)應(yīng)的異常處理方法。當(dāng)控制器中發(fā)生指定類型的異常時(shí),SpringBoot會(huì)自動(dòng)調(diào)用對(duì)應(yīng)的異常處理方法,并將異常對(duì)象傳遞給該方法作為參數(shù)。
一般的全局異常處理類
@Slf4j @RestControllerAdvice public class GlobalExceptionHandler { private static final String DEFAULT_ERROR_MESSAGE = "系統(tǒng)繁忙,請(qǐng)稍后再試"; @ExceptionHandler(value = {SQLIntegrityConstraintViolationException.class}) public DataVO handleSQLIntegrityConstraintViolationException(SQLIntegrityConstraintViolationException e) { log.error(e.getMessage(), e); return new DataVO(SysConstant.CODE_ERROR, "當(dāng)前主鍵存在于其他表的外鍵約束,請(qǐng)先處理父表中的該外鍵"); } @ExceptionHandler(value = {FileNotFoundException.class}) public DataVO handleFileNotFoundException(FileNotFoundException e) { log.error(e.getMessage(), e); return new DataVO(SysConstant.CODE_ERROR, "路徑不存在"); } @ExceptionHandler(value = {Exception.class}) public DataVO handleException(Exception e) { log.error(e.getMessage(), e); return new DataVO(SysConstant.CODE_ERROR, DEFAULT_ERROR_MESSAGE); } }
如例所示,使用@RestControllerAdvice注解聲明了GlobalExceptionHandler作為全局異常處理類,在這個(gè)類中的方法使用@ExceptionHandler注解指定某個(gè)方法處理對(duì)應(yīng)的錯(cuò)誤類型。
在這個(gè)一般的全局處理類中,@ExceptionHandler注解會(huì)根據(jù)異常類型選擇最精確的處理方法進(jìn)行處理,如果沒有找到對(duì)應(yīng)的處理方法,則會(huì)選擇更加通用的處理方法。這樣的處理方式會(huì)導(dǎo)致代碼重復(fù),每個(gè)處理方法都需要寫一遍通用的操作,例如日志記錄。
改進(jìn)全局異常處理類
為了解決這個(gè)問題,我們可以將通用的異常處理邏輯抽象到一個(gè)方法中,并在處理方法中調(diào)用該方法
@Slf4j @RestControllerAdvice public class GlobalExceptionHandler { private static final String DEFAULT_ERROR_MESSAGE = "系統(tǒng)繁忙,請(qǐng)稍后再試"; @ExceptionHandler(value = {SQLIntegrityConstraintViolationException.class}) public DataVO handleSQLIntegrityConstraintViolationException(SQLIntegrityConstraintViolationException e) { log.error(e.getMessage(), e); return handleException(DEFAULT_ERROR_MESSAGE, e); } @ExceptionHandler(value = {FileNotFoundException.class}) public DataVO handleFileNotFoundException(FileNotFoundException e) { log.error(e.getMessage(), e); return handleException(DEFAULT_ERROR_MESSAGE, e); } @ExceptionHandler(value = {Exception.class}) public DataVO handleException(Exception e) { log.error(e.getMessage(), e); return handleException(DEFAULT_ERROR_MESSAGE, e); } private DataVO handleException(String defaultMessage, Throwable e) { if (e instanceof BusinessException) { return new DataVO(SysConstant.CODE_ERROR, e.getMessage()); } else if (e instanceof MethodArgumentNotValidException) { return new DataVO(SysConstant.CODE_ERROR, ((MethodArgumentNotValidException) e).getBindingResult().getFieldError().getDefaultMessage()); } else { return new DataVO(SysConstant.CODE_ERROR, defaultMessage); } } }
在這個(gè)示例中,定義了一個(gè)私有方法handleException,該方法接受一個(gè)默認(rèn)錯(cuò)誤消息和一個(gè)Throwable對(duì)象作為參數(shù),并返回一個(gè)DataVO對(duì)象。
在處理方法中,調(diào)用handleException方法來處理異常,并將默認(rèn)錯(cuò)誤消息作為參數(shù)傳遞給handleException方法。
這樣,我們就可以將通用的異常處理邏輯抽象到一個(gè)方法中,避免了重復(fù)代碼。同時(shí),我們也可以在handleException方法中添加自己的邏輯,例如記錄日志等。
優(yōu)雅的全局異常處理類
handleException方法中的if-else語句太過臃腫,可以使用Map來優(yōu)化這個(gè)方法
@Slf4j @RestControllerAdvice public class GlobalExceptionHandler { private static final String errorMsg = SysConstant.DEFAULT_ERROR; private static final Map<Class<? extends Throwable>, Function<Throwable, DataVO<Null>>> EXCEPTION_HANDLER_MAP = new HashMap<>(); static { EXCEPTION_HANDLER_MAP.put(RuntimeException.class, e -> new DataVO<>(SysConstant.CODE_ERROR, e.getMessage())); EXCEPTION_HANDLER_MAP.put(DataIntegrityViolationException.class, e -> new DataVO<>(SysConstant.CODE_ERROR, "當(dāng)前主鍵存在于其他表的外鍵約束,請(qǐng)先處理父表中的該外鍵")); EXCEPTION_HANDLER_MAP.put(FileNotFoundException.class, e -> new DataVO<>(SysConstant.CODE_ERROR, "路徑不存在")); } @ExceptionHandler(Exception.class) public DataVO<Null> handleException(Exception e) { log.error(e.toString()); return EXCEPTION_HANDLER_MAP.getOrDefault(e.getClass(), t -> new DataVO<>(SysConstant.CODE_ERROR, errorMsg)).apply(e); } }
在這個(gè)示例中,使用Map來存儲(chǔ)異常處理函數(shù),每個(gè)函數(shù)都接受一個(gè)Throwable對(duì)象作為參數(shù),并返回一個(gè)DataVO對(duì)象。
在處理方法中,根據(jù)異常類型從Map中獲取對(duì)應(yīng)的處理函數(shù),如果沒有找到對(duì)應(yīng)的處理函數(shù),則使用默認(rèn)的處理函數(shù)。
然后,調(diào)用獲取到的處理函數(shù)來處理異常。這樣,就可以將異常處理函數(shù)集中在一個(gè)Map中,避免了if-else語句的臃腫。同時(shí),也可以在異常處理函數(shù)handleException中添加自己的邏輯,例如記錄日志等。
我的通用值對(duì)象類DataVO(Data Value Object)
@Data //@JsonPropertyOrder({"code","msg","count","data"})//指定返回給前端的字段順序 public class DataVO<T> { private Integer code = 0; private String msg = ""; private Long count; private List<T> data; public DataVO() { } public DataVO(Integer code, String msg) { this.code = code; this.msg = msg; } public DataVO(Integer code, String msg, Long count, List<T> data) { this.code = code; this.msg = msg; this.count = count; this.data = data; } }
遇到問題
在測(cè)試的時(shí)候要處理外鍵約束報(bào)錯(cuò)問題,控制臺(tái)發(fā)現(xiàn)報(bào)錯(cuò)原因是SQLIntegrityConstraintViolationException,于是我在靜態(tài)異常Map中加入了SQLIntegrityConstraintViolationException處理,但是發(fā)現(xiàn)并沒有正常處理,還是返回的默認(rèn)異常信息。
于是在控制臺(tái)輸出了一下那個(gè)錯(cuò)誤的類,e.getClass()發(fā)現(xiàn)實(shí)際上是DataIntegrityViolationException,SQLIntegrityConstraintViolationException只是cause,改完之后測(cè)試成功處理異常并返回特定異常信息給前端。
到此這篇關(guān)于SpringBoot中@RestControllerAdvice注解全局異常處理類的文章就介紹到這了,更多相關(guān)@RestControllerAdvice注解全局異常內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- springboot的統(tǒng)一異常處理,使用@RestControllerAdvice詳解
- SpringBoot項(xiàng)目中@RestControllerAdvice全局異常失效問題的解決
- SpringBoot中@RestControllerAdvice @ExceptionHandler異常統(tǒng)一處理類失效原因分析
- SpringBoot中@RestControllerAdvice注解的使用
- SpringBoot的@RestControllerAdvice作用詳解
- SpringBoot常用注解@RestControllerAdvice詳解
- SpringBoot中的@RestControllerAdvice注解詳解
- SpringBoot?@RestControllerAdvice注解對(duì)返回值統(tǒng)一封裝的處理方法
- SpringBoot中@RestControllerAdvice 全局異常處理的實(shí)現(xiàn)
相關(guān)文章
request如何獲取完整url(包括域名、端口、參數(shù))
這篇文章主要介紹了request如何獲取完整url(包括域名、端口、參數(shù))問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-12-12Spring?Security認(rèn)證器實(shí)現(xiàn)過程詳解
一些權(quán)限框架一般都包含認(rèn)證器和決策器,前者處理登陸驗(yàn)證,后者處理訪問資源的控制,這篇文章主要介紹了Spring?Security認(rèn)證器實(shí)現(xiàn)過程,需要的朋友可以參考下2022-06-06Jsoup獲取全國(guó)地區(qū)數(shù)據(jù)屬性值(省市縣鎮(zhèn)村)
這篇文章主要介紹了Jsoup獲取全國(guó)地區(qū)數(shù)據(jù)屬性值(省市縣鎮(zhèn)村)的相關(guān)資料,需要的朋友可以參考下2015-10-10Spring Boot 在啟動(dòng)時(shí)進(jìn)行配置文件加解密的方法詳解
這篇文章主要介紹了Spring Boot 在啟動(dòng)時(shí)進(jìn)行配置文件加解密的方法,本文通過實(shí)例給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-06-06Java之多個(gè)線程順序循環(huán)執(zhí)行的幾種實(shí)現(xiàn)
這篇文章主要介紹了Java之多個(gè)線程順序循環(huán)執(zhí)行的幾種實(shí)現(xiàn)方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-09-09Java JTable 實(shí)現(xiàn)日歷的示例
這篇文章主要介紹了Java JTable 實(shí)現(xiàn)日歷的示例,幫助大家更好的理解和學(xué)習(xí)Java jtable的使用方法,感興趣的朋友可以了解下2020-10-10