SpringBoot參數(shù)校驗的最佳實戰(zhàn)教程
前言
我們這里使用hibernate-validator作為對象參數(shù)驗證器,所以在正式介紹SpringBoot參數(shù)驗證之前,需要先簡單了解一下hibernate-validator的使用。
hibernate-validator基本使用
引入依賴
<dependency>
<groupId>org.hibernate</groupId>
<artifactId>hibernate-validator</artifactId>
<version>6.0.17.Final</version>
</dependency>
<dependency>
<groupId>javax.validation</groupId>
<artifactId>validation-api</artifactId>
</dependency>
編寫需要驗證對象
驗證要求 person對象的用戶名不能為空,年齡在1-150歲之間。
@Data
public class Person {
@NotBlank(message = "username must not be null")
private String username;
@Min(value = 1, message = "age must be >= 1")
@Max(value = 150, message = "age must be < 150")
private Integer age;
}
驗證對象屬性是否符合要求
/**
* 對象驗證器
*/
public Validator validator() {
ValidatorFactory validatorFactory =
Validation
.byProvider(HibernateValidator.class)
.configure()
// 驗證屬性時,如果有一個驗證不通過就返回,不需要驗證所有屬性
.addProperty("hibernate.validator.fail_fast", "true")
.buildValidatorFactory();
return validatorFactory.getValidator();
}
@Test
public void test() throws Exception {
Person person = new Person();
Set<ConstraintViolation<Person>> validate = validator().validate(person);
validate.forEach(errorParam -> {
System.out.println(errorParam.getMessage());
});
}
- 我們只需要驗證的對象實例即可完成對象驗證,如果驗證成功,那么返回一個空的集合,如果驗證失敗,會返回具體的驗證失敗的屬性信息。
- 我們輸出驗證失敗的錯誤信息如下:
username must not be null
驗證規(guī)則
validator提供了大量的驗證注解供我們使用,主要以下幾類:

空/非空驗證
- @Null 元素必須為空
- @NotNull 元素不能為空,空字符串""是非空
以下所有驗證規(guī)則都在元素非空的時候才會進(jìn)行驗證,如果傳入的元素為空,驗證都會通過。
bool
- @AssertTrue 元素必須為true
- @AssertFalse 元素必須為false
時間
- @Future 元素必須是未來的某個時間。
- @FutureOrPresent 元素必須是未來或者現(xiàn)在的某個時間。
- @Past 元素必須是過去的某個時間。
- @PastOrPresent 元素必須是過去或者現(xiàn)在的某個時間。
數(shù)學(xué)
數(shù)字類型可以是BigDecimal、BigInteger、CharSequence 、byte 、 short 、 int 、 long以及它們各自的包裝器類型
- @Digits 元素必須是該數(shù)字類型下可以被接受的數(shù)值范圍內(nèi)。
- @Negative 元素必須是負(fù)數(shù)
- @NegativeOrZero 元素必須小于等于0
- @Positive 元素必須大于0
- @PositiveOrZero 元素必須大于等于0
- @Max,@Min 元素的大小必須符合指定大小
字符串
- @Email 郵箱格式驗證
- @NotBlack 驗證字符串非空,空字符串""也屬于空
- @Pattern 字符串正則驗證
模板正則
validator提供了字符串模板正則的注解,這里提供一份常用的正則表達(dá)式,大家可以直接作為常量工具類放到項目里使用
public interface ValidatorPattern {
/**
* 正則表達(dá)式:驗證用戶名
* 1.長度在5-17
* 2.由大寫小寫字母構(gòu)成
*/
String REGEX_USERNAME = "^[a-zA-Z]\w{5,17}$";
/**
* 正則表達(dá)式:驗證密碼
* 密碼只能為 6 - 12位數(shù)字,字母及常用符號組成。
*/
String REGEX_PASSWORD = "^(?=.*[a-zA-Z])(?=.*[0-9])[A-Za-z0-9._~!@#$^&*]{6,12}$";
/**
* 正則表達(dá)式:驗證手機號
*/
String REGEX_MOBILE = "^[1][34578]\d{9}$";
/**
* 正則表達(dá)式:驗證郵箱
*/
String REGEX_EMAIL = "^.+@([a-z0-9A-Z]+(-[a-z0-9A-Z]+)?\.)+[a-zA-Z]{2,}$";
/**
* 正則表達(dá)式:驗證漢字
*/
String REGEX_CHINESE = "^[\u4e00-\u9fa5],*$";
/**
* 正則表達(dá)式:驗證身份證
*/
String REGEX_ID_CARD = "(^\d{18}$)|(^\d{15}$)";
/**
* 正則表達(dá)式:驗證URL
*/
String REGEX_URL = "http(s)?://([\w-]+\.)+[\w-]+(/[\w- ./?%&=]*)?";
/**
* 正則表達(dá)式:驗證IP地址
*/
String REGEX_IP_ADDR = "(25[0-5]|2[0-4]\d|[0-1]\d{2}|[1-9]?\d)";
/**
* 車牌號正則
*/
String LICENSE_NO = "^[京津滬渝冀豫云遼黑湘皖魯新蘇浙贛鄂桂甘晉蒙陜吉閩貴粵青藏川寧瓊使領(lǐng)A-Z][A-Z][A-Z0-9]{4,5}[A-Z0-9掛學(xué)警港澳]$";
/**
* 姓名校驗
* 1~15位
* 姓名支持空格和中文的點
*/
String NAME = "[\u4e00-\u9fa5\u00b7\sA-Za-z]{1,15}$";
/**
* 表情正則
*/
String EMOJI = "[\ud83c\udc00-\ud83c\udfff]|[\ud83d\udc00-\ud83d\udfff]|[\u2600-\u27ff]";
/**
* 數(shù)字正則
*/
String NUMBER = "^[0-9]*$";
/**
* n位的數(shù)字
*/
String N_NUMS = "^\d{n}$";
}
SpringBoot整合hibernate-validator
引入依賴
這個不再贅述,直接拷貝上文的依賴信息
配置hibernate-validator驗證器對象
在配置類中加入hibernate-validator驗證器對象
@Bean
@Primary
public Validator validator() {
ValidatorFactory validatorFactory =
Validation
.byProvider(HibernateValidator.class)
.configure()
.addProperty("hibernate.validator.fail_fast", "true")
.buildValidatorFactory();
return validatorFactory.getValidator();
}
借助SpringMVC統(tǒng)一異常處理處理參數(shù)校驗結(jié)果
配置好后,Spring會自動幫助我們進(jìn)行參數(shù)驗證,如果參數(shù)驗證不通過,會拋出BindException異常,我們剛剛手動驗證時的Set<ConstraintViolation<Person>>通過該異常獲取。
我們這可以通過借助SpringMVC統(tǒng)一異常處理的能力處理這個異常
@Slf4j
@RestControllerAdvice
public class BaseExceptionHandler {
/**
* spring validation 自動校驗的參數(shù)異常
*
* @param e BindException
* @return R<Void>
*/
@ResponseStatus(org.springframework.http.HttpStatus.PAYMENT_REQUIRED)
@ExceptionHandler(BindException.class)
public R<Void> handler(BindException e) {
String defaultMsg = e.getBindingResult().getAllErrors()
.stream()
.map(ObjectError::getDefaultMessage)
.collect(Collectors.joining(":"));
log.warn(defaultMsg);
return R.of(IRespCode.PARAMETERS_ANOMALIES.getCode(), e.getMessage());
}
}
使用參數(shù)校驗
我們只需要在校驗參數(shù)的方法傳參上標(biāo)注@Valid或者@Validated都行
@PostMapping("register")
public R<Void> register(@Valid @RequestBody Person person) {
// todo
return R.ok();
}
分組校驗
那么@Valid和@Validated有什么區(qū)別呢?
Validated比Valid多了一個屬性,這個屬性用于分組校驗使用
public @interface Valid {
}
public @interface Validated {
Class<?>[] value() default {};
}
啥叫分組校驗?
就是一個實體類中的屬性,在不同的方法傳參中,方法的對屬性的要求不同。
比如說,Person類中有三個屬性,一個是用戶名稱,一個是郵箱,一個是年齡。
在注冊用戶接口中,用戶名稱,郵箱和年齡都不能為空,但是在更改用戶的信息接口中,用戶的年齡和郵箱都可以為空,但是用戶名稱不能為空。
這時候,我們就可以按照對屬性校驗的要求進(jìn)行分組。
新建一個RegisterGroup分組,該分組只是一個空的接口,僅僅用于標(biāo)記該校驗要求
public interface RegisterGroup {
}
對校驗要求進(jìn)行分組
@Data
public class Person {
@NotBlank(message = "username must not be null")
private String username;
@Min(value = 1, message = "age must be >= 1")
@Max(value = 150, message = "age must be < 150")
@NotNull(message = "age must not be null", groups = RegisterGroup.class)
private Integer age;
@Email(message = "email format error")
@NotBlank(message = "email must not be null",groups = RegisterGroup.class)
private String email;
}
方法調(diào)用時,加入分組要求
@PostMapping("register")
public R<Void> register(@Validated(value = RegisterGroup.class) @RequestBody Person person) {
// todo
return R.ok();
}
這種方式其實不推薦使用,我在標(biāo)題的時候,也已經(jīng)標(biāo)記為“過時”,因為,我們完全可以為這兩個不同的接口創(chuàng)建兩個不同的實體類,而不是使用分組對校驗要求進(jìn)行隔離,因為實際生產(chǎn)環(huán)境中,分組可能有非常多個,這會為我們的程序的可讀性埋下隱患,后期開發(fā)人員難以維護,而且對于自動生成API文檔也不友好。大家對于分組只需要了解即可,不建議在項目開發(fā)中使用。
總結(jié)
到此這篇關(guān)于SpringBoot參數(shù)校驗的文章就介紹到這了,更多相關(guān)SpringBoot參數(shù)校驗內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
SpringBoot中整合Ehcache實現(xiàn)熱點數(shù)據(jù)緩存的詳細(xì)過程
這篇文章主要介紹了SpringBoot中整合Ehcache實現(xiàn)熱點數(shù)據(jù)緩存,SpringBoot 中使用 Ehcache 比較簡單,只需要簡單配置,說白了還是 Spring Cache 的用法,合理使用緩存機制,可以很好地提高項目的響應(yīng)速度,需要的朋友可以參考下2023-04-04
Java中@ConfigurationProperties實現(xiàn)自定義配置綁定問題分析
這篇文章主要介紹了@ConfigurationProperties實現(xiàn)自定義配置綁定問題,本文通過實例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2021-08-08

