spring使用validation參數(shù)及全局異常檢測方式
1.validation參數(shù)驗(yàn)證工具
1.1.validation-api技術(shù)鏈
validation-api是一個(gè)Java的數(shù)據(jù)校驗(yàn)規(guī)范,它定義了一套用于校驗(yàn)Java Bean的API。它是JSR 303規(guī)范的一部分,也被稱為Bean Validation。validation-api提供了一系列的注解,用于在Java類的字段、方法參數(shù)和方法返回值上添加校驗(yàn)規(guī)則。這些注解包括@NotNull、@Size、@Min、@Max等等,每個(gè)注解都有特定的校驗(yàn)規(guī)則,用于驗(yàn)證數(shù)據(jù)是否滿足特定的條件。
Hibernate Validator是validation-api的唯一實(shí)現(xiàn)。除了提供注解和校驗(yàn)規(guī)則,Hibernate Validator還提供了一系列的工具類和接口,用于處理校驗(yàn)結(jié)果和錯(cuò)誤信息。它可以將校驗(yàn)結(jié)果封裝為一個(gè)Validator對(duì)象,并提供了各種方法來獲取校驗(yàn)結(jié)果、錯(cuò)誤信息和錯(cuò)誤類型。Hibernate Validator還支持國際化,可以根據(jù)不同的語言環(huán)境顯示相應(yīng)的錯(cuò)誤信息。它還提供了可自定義的錯(cuò)誤消息模板,可以根據(jù)實(shí)際需求來設(shè)置錯(cuò)誤消息的格式和內(nèi)容。
spring-boot-starter-validation是一個(gè)Spring Boot的starter,用于集成和簡化Spring框架中的數(shù)據(jù)校驗(yàn)功能。spring-boot-starter-validation通過自動(dòng)配置的方式,將Hibernate Validator集成到了Spring Boot的應(yīng)用中,從而簡化了數(shù)據(jù)校驗(yàn)的配置過程。通過引入這個(gè)starter,我們可以在應(yīng)用中使用標(biāo)準(zhǔn)的注解來進(jìn)行數(shù)據(jù)校驗(yàn),而不需要手動(dòng)配置和引入相關(guān)的依賴項(xiàng)。
1.2.validation常用注解
validation-api提供了一些常用的注解
| @NotNull | 用于校驗(yàn)字段或方法參數(shù)的值不能為空 |
| @NotEmpty | 用于校驗(yàn)字符串、集合或數(shù)組的值不能為空,且長度不能為0 |
| @NotBlank | 用于校驗(yàn)字符串的值不能為空或只包含空格 |
| @Size | 用于校驗(yàn)字符串、集合或數(shù)組的長度是否在指定范圍內(nèi) |
| @Min | 用于校驗(yàn)數(shù)字的值是否大于等于指定的最小值 |
| @Pattern | 用于校驗(yàn)字符串的值是否匹配指定的正則表達(dá)式 |
| 用于校驗(yàn)字符串的值是否符合Email格式 | |
| @DecimalMin | 用于校驗(yàn)數(shù)字的值是否大于等于指定的最小值,可以指定最小值的精度 |
Hibernate Validator 提供了一些拓展注解,用于在 validation-api 注解基礎(chǔ)上實(shí)現(xiàn)更復(fù)雜的數(shù)據(jù)校驗(yàn)。
以下是一些常見的 Hibernate Validator 拓展注解(部分)
| @URL | 用于校驗(yàn)字符串的值是否符合 URL 地址格式 |
| @CreditCardNumber | 用于校驗(yàn)信用卡號(hào)的格式是否正確 |
| @Currency | 用于校驗(yàn)貨幣代碼是否正確 |
| @ISBN | 用于校驗(yàn)國際標(biāo)準(zhǔn)書號(hào)(ISBN)是否正確 |
| @Range | 用于校驗(yàn)數(shù)字的值是否在指定范圍內(nèi) |
1.3.代碼示例
對(duì)于一個(gè)請(qǐng)求參數(shù),根據(jù)需求進(jìn)行注解申明
import jakarta.validation.constraints.DecimalMax;
import jakarta.validation.constraints.DecimalMin;
import jakarta.validation.constraints.NotEmpty;
import jakarta.validation.constraints.Size;
import lombok.Data;
@Data
public class ReqCreateCatalog {
@NotEmpty
private String gameId;
@DecimalMin(value = "1")
@DecimalMax(value = "2")
private int type;
//目標(biāo)目錄
@Size(max = 50, message = "名稱長度不能超過{max}個(gè)字符")
private String catalog;
}controller方法參數(shù)增加@Valid注解
@PostMapping(value = "/createCatalog")
public Response create(@Valid @RequestBody ReqCreateCatalog req) {
}如果請(qǐng)求故意把gameId參數(shù)設(shè)為空,

則程序輸出異常日志

2. 全局異常處理器
2.1.使用@RestControllerAdvice攔截異常
上面的例子可以看出,當(dāng)驗(yàn)證器拋出MethodArgumentNotValidException異常,如果沒有一個(gè)全局異常捕獲器對(duì)其進(jìn)行捕獲并進(jìn)行轉(zhuǎn)義,前端得到的響應(yīng)很奇怪。
SpringMVC使用@RestControllerAdvice注解對(duì)全局異常進(jìn)行攔截,例如下面的代碼
(也可以使用@ControllerAdvice注解,但每個(gè)方法還需要添加@ResponseBody注解)
@Slf4j
@RestControllerAdvice
public class GlobalExceptionHandler {
@ExceptionHandler(MethodArgumentNotValidException.class)
public Response<Object> handleMethodArgumentNotValidException(MethodArgumentNotValidException e, HttpServletRequest request) {
String requestURI = request.getRequestURI();
log.error("請(qǐng)求地址'{}',參數(shù)校驗(yàn)失敗'{}'", requestURI, e.getMessage());
return Response.fail(I18nConstants.COMMON_ILLEGAL_PARAMS);
}
@ExceptionHandler(BusinessRequestException.class)
public Response<Object> handleBusinessRequestException(BusinessRequestException e, HttpServletRequest request) {
String requestURI = request.getRequestURI();
log.error("請(qǐng)求地址'{}',權(quán)限碼校驗(yàn)失敗'{}'", requestURI, e.getMessage());
return Response.fail(e.getErrorCode());
}
@ExceptionHandler(value = Exception.class)
public Response<Object> handleException(Exception e, HttpServletRequest request) {
LoggerUtil.error("", e);
return Response.fail(I18nConstants.COMMON_INTERNAL_ERROR);
}
}2.2.異常攔截方法有以下幾個(gè)注意地方
- 方法名必須以handle開頭;
- 需要添加@ExceptionHandler注解,注解的值是一個(gè)異常類數(shù)組,當(dāng)value為空,則默認(rèn)取方法的異常參數(shù)類型;
- 當(dāng)程序觸發(fā)了一個(gè)異常,如果在全局異常攔截器找不到映射,則會(huì)遞歸找它的父異常,則到父類Throwalbe為止;
- 如果多個(gè)@RestControllerAdvice示例綁定了同一個(gè)異常,程序不會(huì)報(bào)錯(cuò),但只有一個(gè)攔截器起作用。
2.3.使用案例
使用全局異常攔截器,重新發(fā)起請(qǐng)求,可以看到客戶端輸出:
{
"success": false,
"message": 1003, //I18nConstants.COMMON_ILLEGAL_PARAMS
"data": null
}總結(jié)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
Mybatis數(shù)據(jù)批量插入如何實(shí)現(xiàn)
這篇文章主要介紹了Mybatis數(shù)據(jù)批量插入如何實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-07-07
JAVA實(shí)現(xiàn)生成順序ID,不浪費(fèi)ID
這篇文章主要介紹了JAVA實(shí)現(xiàn)生成順序ID,不浪費(fèi)ID問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-04-04
Java中小球碰撞并使用按鈕控制數(shù)量實(shí)例代碼
這篇文章主要給大家介紹了關(guān)于Java中小球碰撞并使用按鈕控制數(shù)量的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2021-12-12
Java結(jié)合redistemplate使用分布式鎖案例講解
在Java中使用RedisTemplate結(jié)合Redis來實(shí)現(xiàn)分布式鎖是一種常見的做法,特別適用于微服務(wù)架構(gòu)或多實(shí)例部署的應(yīng)用程序中,以確保數(shù)據(jù)的一致性和避免競態(tài)條件,下面給大家分享使用Spring Boot和RedisTemplate實(shí)現(xiàn)分布式鎖的案例,感興趣的朋友一起看看吧2024-08-08
如何在Intellij中安裝LeetCode刷題插件方便Java刷題
這篇文章主要介紹了如何在Intellij中安裝LeetCode刷題插件方便Java刷題,本文通過圖文并茂的形式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-08-08
解決mybatis 執(zhí)行mapper的方法時(shí)報(bào)空指針問題
這篇文章主要介紹了解決mybatis 執(zhí)行mapper的方法時(shí)報(bào)空指針問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-07-07

