SpringMVC實現(xiàn)全局異常處理器的經(jīng)典案例
通過 @ControllerAdvice 注解,我們可以在一個地方對所有 @Controller 注解的控制器進(jìn)行管理。
注解了 @ControllerAdvice 的類的方法可以使用 @ExceptionHandler、 @InitBinder、 @ModelAttribute 注解到方法上,這對所有注解了 @RequestMapping 的控制器內(nèi)的方法都有效。
@ExceptionHandler:用于捕獲所有控制器里面的異常,并進(jìn)行處理。@InitBinder:用來設(shè)置 WebDataBinder,WebDataBinder 用來自動綁定前臺請求參數(shù)到 Model 中。@ModelAttribute:@ModelAttribute 本來的作用是綁定鍵值對到 Model
里,此處是讓全局的@RequestMapping 都能獲得在此處設(shè)置的鍵值對。
本文使用 @ControllerAdvice + @ExceptionHandler 進(jìn)行全局的 Controller 層異常處理。只要設(shè)計得當(dāng),就再也不用在 Controller 層進(jìn)行 try-catch 了!
一、經(jīng)典案例
需求:希望通過全局統(tǒng)一的異常處理將自定義錯誤碼以json的形式發(fā)送給前端。
1、統(tǒng)一返回結(jié)果類 ApiResult
首先,定義一個統(tǒng)一結(jié)果返回類,最終需要將這個結(jié)果類的內(nèi)容返回給前端。
package com.tao.smp.exception;
/**
* Api統(tǒng)一的返回結(jié)果類
*/
public class ApiResult {
/**
* 結(jié)果碼
*/
private String code;
/**
* 結(jié)果碼描述
*/
private String msg;
public ApiResult() {
}
public ApiResult(ResultCode resultCode) {
this.code = resultCode.getCode();
this.msg = resultCode.getMsg();
}
/**
* 生成一個ApiResult對象, 并返回
*
* @param resultCode
* @return
*/
public static ApiResult of(ResultCode resultCode) {
return new ApiResult(resultCode);
}
public String getCode() {
return code;
}
public void setCode(String code) {
this.code = code;
}
public String getMsg() {
return msg;
}
public void setMsg(String msg) {
this.msg = msg;
}
@Override
public String toString() {
return "ApiResult{" +
"code='" + code + '\'' +
", msg='" + msg + '\'' +
'}';
}
}2、錯誤碼枚舉類 ResultCode
有了 ApiResult ,接下來需要定義一個枚舉類, 來包含所有自定義的結(jié)果碼。
package com.tao.smp.exception;
/**
* 錯誤碼
*/
public enum ResultCode {
/**
* 成功
*/
SUCCESS("0", "success"),
/**
* 未知錯誤
*/
UNKNOWN_ERROR("0x10001", "unkonwn error"),
/**
* 用戶名錯誤或不存在
*/
USERNAME_ERROR("0x10002", "username error or does not exist"),
/**
* 密碼錯誤
*/
PASSWORD_ERROR("0x10003", "password error"),
/**
* 用戶名不能為空
*/
USERNAME_EMPTY("0x10004", "username can not be empty");
/**
* 結(jié)果碼
*/
private String code;
/**
* 結(jié)果碼描述
*/
private String msg;
ResultCode(String code, String msg) {
this.code = code;
this.msg = msg;
}
public String getCode() {
return code;
}
public String getMsg() {
return msg;
}
}3、自定義業(yè)務(wù)異常類 BusinessRuntimeException
接下來需要定義我們自己的業(yè)務(wù)異常類,以后和業(yè)務(wù)相關(guān)的異常通通拋出這個異常類,我們將錯誤碼枚舉變量的值存于其中。
package com.tao.smp.exception;
/**
* 自定義業(yè)務(wù)異常
*/
public class BusinessRuntimeException extends RuntimeException {
/**
* 結(jié)果碼
*/
private String code;
/**
* 結(jié)果碼描述
*/
private String msg;
/**
* 結(jié)果碼枚舉
*/
private ResultCode resultCode;
public BusinessRuntimeException(ResultCode resultCode) {
super(resultCode.getMsg());
this.code = resultCode.getCode();
this.msg = resultCode.getMsg();
this.resultCode = resultCode;
}
public String getCode() {
return code;
}
public void setCode(String code) {
this.code = code;
}
public String getMsg() {
return msg;
}
public void setMsg(String msg) {
this.msg = msg;
}
public ResultCode getResultCode() {
return resultCode;
}
public void setResultCode(ResultCode resultCode) {
this.resultCode = resultCode;
}
}4、全局異常處理類 GlobalExceptionResolver
最后便是定義全局異常處理類。
- 通過 @ControllerAdvice 指定該類為 Controller 增強(qiáng)類。
- 通過 @ExceptionHandler 自定捕獲的異常類型。
- 通過 @ResponseBody 返回 json 到前端。
注意一點:被@ControllerAdvice注解的全局異常處理類也是一個 Controller ,我們需要配置掃描路徑,確保能夠掃描到這個Controller。
package com.tao.smp.exception;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.web.bind.annotation.ControllerAdvice;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.bind.annotation.ResponseBody;
/**
* 全局Controller層異常處理類
*/
@ControllerAdvice
public class GlobalExceptionResolver {
private static final Logger LOG = LoggerFactory.getLogger(GlobalExceptionResolver.class);
/**
* 處理所有不可知異常
*
* @param e 異常
* @return json結(jié)果
*/
@ExceptionHandler(Exception.class)
@ResponseBody
public ApiResult handleException(Exception e) {
// 打印異常堆棧信息
LOG.error(e.getMessage(), e);
return ApiResult.of(ResultCode.UNKNOWN_ERROR);
}
/**
* 處理所有業(yè)務(wù)異常
*
* @param e 業(yè)務(wù)異常
* @return json結(jié)果
*/
@ExceptionHandler(BusinessRuntimeException.class)
@ResponseBody
public ApiResult handleOpdRuntimeException(BusinessRuntimeException e) {
// 不打印異常堆棧信息
LOG.error(e.getMsg());
return ApiResult.of(e.getResultCode());
}
}二、測試
1、測試 TestController
package com.tao.smp.controller;
import com.tao.smp.exception.BusinessRuntimeException;
import com.tao.smp.exception.ResultCode;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
/**
* 測試異常的拋出
*/
@Controller
@RequestMapping("/")
public class TestController {
/**
* 測試返回異常信息
* @return
*/
@GetMapping("/exception")
public String returnExceptionInfo() {
if (1 != 2) {
// 用戶民錯誤或不存在異常
throw new BusinessRuntimeException(ResultCode.USERNAME_ERROR);
}
return "success";
}
}


到此這篇關(guān)于SpringMVC實現(xiàn)全局異常處理器的文章就介紹到這了,更多相關(guān)SpringMVC全局異常處理器內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Spring Boot 實現(xiàn)https ssl免密登錄(X.509 pki登錄)
這篇文章主要介紹了Spring Boot 實現(xiàn)https ssl免密登錄(X.509 pki登錄),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-01-01
SpringBoot整合Mybatis-Plus+Druid實現(xiàn)多數(shù)據(jù)源配置功能
本文主要講解springboot?+mybatisplus?+?druid?實現(xiàn)多數(shù)據(jù)源配置功能以及一些必要的準(zhǔn)備及代碼說明,具有一定的參考價值,感興趣的小伙伴可以借鑒一下2023-06-06
Spring整合多數(shù)據(jù)源實現(xiàn)動態(tài)切換的實例講解
下面小編就為大家?guī)硪黄猄pring整合多數(shù)據(jù)源實現(xiàn)動態(tài)切換的實例講解。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-07-07
java 驗證用戶是否已經(jīng)登錄與實現(xiàn)自動登錄方法詳解
本文主要介紹了java 驗證用戶是否已經(jīng)登錄與實現(xiàn)自動登錄的方法。具有一定的參考價值,下面跟著小編一起來看下吧2017-01-01
java中CompleteFuture與Future的區(qū)別小結(jié)
本文主要介紹了java中CompleteFuture與Future的區(qū)別小結(jié),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-12-12

