SpringMVC實(shí)現(xiàn)全局異常處理器的經(jīng)典案例
通過(guò) @ControllerAdvice 注解,我們可以在一個(gè)地方對(duì)所有 @Controller 注解的控制器進(jìn)行管理。
注解了 @ControllerAdvice 的類(lèi)的方法可以使用 @ExceptionHandler、 @InitBinder、 @ModelAttribute 注解到方法上,這對(duì)所有注解了 @RequestMapping 的控制器內(nèi)的方法都有效。
@ExceptionHandler:用于捕獲所有控制器里面的異常,并進(jìn)行處理。@InitBinder:用來(lái)設(shè)置 WebDataBinder,WebDataBinder 用來(lái)自動(dòng)綁定前臺(tái)請(qǐng)求參數(shù)到 Model 中。@ModelAttribute:@ModelAttribute 本來(lái)的作用是綁定鍵值對(duì)到 Model
里,此處是讓全局的@RequestMapping 都能獲得在此處設(shè)置的鍵值對(duì)。
本文使用 @ControllerAdvice + @ExceptionHandler 進(jìn)行全局的 Controller 層異常處理。只要設(shè)計(jì)得當(dāng),就再也不用在 Controller 層進(jìn)行 try-catch 了!
一、經(jīng)典案例
需求:希望通過(guò)全局統(tǒng)一的異常處理將自定義錯(cuò)誤碼以json的形式發(fā)送給前端。
1、統(tǒng)一返回結(jié)果類(lèi) ApiResult
首先,定義一個(gè)統(tǒng)一結(jié)果返回類(lèi),最終需要將這個(gè)結(jié)果類(lèi)的內(nèi)容返回給前端。
package com.tao.smp.exception;
/**
* Api統(tǒng)一的返回結(jié)果類(lèi)
*/
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();
}
/**
* 生成一個(gè)ApiResult對(duì)象, 并返回
*
* @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、錯(cuò)誤碼枚舉類(lèi) ResultCode
有了 ApiResult ,接下來(lái)需要定義一個(gè)枚舉類(lèi), 來(lái)包含所有自定義的結(jié)果碼。
package com.tao.smp.exception;
/**
* 錯(cuò)誤碼
*/
public enum ResultCode {
/**
* 成功
*/
SUCCESS("0", "success"),
/**
* 未知錯(cuò)誤
*/
UNKNOWN_ERROR("0x10001", "unkonwn error"),
/**
* 用戶(hù)名錯(cuò)誤或不存在
*/
USERNAME_ERROR("0x10002", "username error or does not exist"),
/**
* 密碼錯(cuò)誤
*/
PASSWORD_ERROR("0x10003", "password error"),
/**
* 用戶(hù)名不能為空
*/
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ù)異常類(lèi) BusinessRuntimeException
接下來(lái)需要定義我們自己的業(yè)務(wù)異常類(lèi),以后和業(yè)務(wù)相關(guān)的異常通通拋出這個(gè)異常類(lèi),我們將錯(cuò)誤碼枚舉變量的值存于其中。
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、全局異常處理類(lèi) GlobalExceptionResolver
最后便是定義全局異常處理類(lèi)。
- 通過(guò) @ControllerAdvice 指定該類(lèi)為 Controller 增強(qiáng)類(lèi)。
- 通過(guò) @ExceptionHandler 自定捕獲的異常類(lèi)型。
- 通過(guò) @ResponseBody 返回 json 到前端。
注意一點(diǎn):被@ControllerAdvice注解的全局異常處理類(lèi)也是一個(gè) Controller ,我們需要配置掃描路徑,確保能夠掃描到這個(gè)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層異常處理類(lèi)
*/
@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());
}
}二、測(cè)試
1、測(cè)試 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;
/**
* 測(cè)試異常的拋出
*/
@Controller
@RequestMapping("/")
public class TestController {
/**
* 測(cè)試返回異常信息
* @return
*/
@GetMapping("/exception")
public String returnExceptionInfo() {
if (1 != 2) {
// 用戶(hù)民錯(cuò)誤或不存在異常
throw new BusinessRuntimeException(ResultCode.USERNAME_ERROR);
}
return "success";
}
}


到此這篇關(guān)于SpringMVC實(shí)現(xiàn)全局異常處理器的文章就介紹到這了,更多相關(guān)SpringMVC全局異常處理器內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Spring Boot 實(shí)現(xiàn)https ssl免密登錄(X.509 pki登錄)
這篇文章主要介紹了Spring Boot 實(shí)現(xiàn)https ssl免密登錄(X.509 pki登錄),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-01-01
Java實(shí)現(xiàn)常見(jiàn)的排序算法代碼實(shí)例
這篇文章主要介紹了Java實(shí)現(xiàn)常見(jiàn)的排序算法代碼實(shí)例,按照思路實(shí)現(xiàn)了以下幾個(gè)排序算法(冒泡排序、直接插入排序、直接選擇排序、快速排序),方便日后用到,特此記錄一下,需要的朋友可以參考下2023-11-11
Java 圖片復(fù)制功能實(shí)現(xiàn)過(guò)程解析
這篇文章主要介紹了Java 圖片復(fù)制功能實(shí)現(xiàn)過(guò)程解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-10-10
SpringBoot整合Mybatis-Plus+Druid實(shí)現(xiàn)多數(shù)據(jù)源配置功能
本文主要講解springboot?+mybatisplus?+?druid?實(shí)現(xiàn)多數(shù)據(jù)源配置功能以及一些必要的準(zhǔn)備及代碼說(shuō)明,具有一定的參考價(jià)值,感興趣的小伙伴可以借鑒一下2023-06-06
Java實(shí)現(xiàn)SHA-1算法實(shí)例
這篇文章主要介紹了Java實(shí)現(xiàn)SHA-1算法,實(shí)例分析了java實(shí)現(xiàn)SHA-1算法的技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-03-03
Java實(shí)現(xiàn)發(fā)送郵件功能時(shí)碰到的坑
之前用163郵箱發(fā)郵件時(shí)明明是成功的,但是使用中國(guó)移動(dòng)自己的郵箱時(shí),無(wú)論如何在linux服務(wù)器中都發(fā)送不成功。下面小編給大家說(shuō)下我是怎么解決的,一起看下吧2016-06-06
Spring整合多數(shù)據(jù)源實(shí)現(xiàn)動(dòng)態(tài)切換的實(shí)例講解
下面小編就為大家?guī)?lái)一篇Spring整合多數(shù)據(jù)源實(shí)現(xiàn)動(dòng)態(tài)切換的實(shí)例講解。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-07-07
java 驗(yàn)證用戶(hù)是否已經(jīng)登錄與實(shí)現(xiàn)自動(dòng)登錄方法詳解
本文主要介紹了java 驗(yàn)證用戶(hù)是否已經(jīng)登錄與實(shí)現(xiàn)自動(dòng)登錄的方法。具有一定的參考價(jià)值,下面跟著小編一起來(lái)看下吧2017-01-01
java中CompleteFuture與Future的區(qū)別小結(jié)
本文主要介紹了java中CompleteFuture與Future的區(qū)別小結(jié),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2023-12-12

