spring-boot-starter-validation?校驗(yàn)參數(shù)的實(shí)現(xiàn)
一、前言
本章介紹使用spring-boot-starter-validation 校驗(yàn) SpringMVC 的入?yún)ⅰ?/p>
<parent> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-parent</artifactId> <version>2.5.2</version> <relativePath/> <!-- lookup parent from repository --> </parent> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-validation</artifactId> </dependency>
二、常用注解

三、定義分組
用于分組校驗(yàn)。
使用場(chǎng)景,對(duì)同一個(gè)對(duì)象例如User(username , id) 在不同的接口時(shí) 需要的校驗(yàn)規(guī)則不同。
例如,訪問一個(gè)接口需要 username 不為null且長(zhǎng)度大于0 ,id>=0 ; 訪問另一個(gè)接口 需要 username 參數(shù)的長(zhǎng)度 在 [1,3]之間。
public class ValidateGroup {
public interface FirstGroup {
}
public interface SecondeGroup {
}
public interface ThirdGroup {
}
}四、定義需要校驗(yàn)的對(duì)象
import javax.validation.constraints.Min;
import javax.validation.constraints.NotEmpty;
import javax.validation.constraints.Size;
import lombok.Data;
@Data
public class User {
@NotEmpty(message = "用戶名不能為空")
@Size(message = "用戶名長(zhǎng)度 [1-3] ", min = 1, max = 3,groups = ValidateGroup.FirstGroup.class)
private String username;
@Min(message = "id不得小于0", value = 0)
private Integer id;
}
五、在handler 即 Controller中 校驗(yàn)
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import com.nbpicc.controller.ValidateGroup.FirstGroup;
@RestController
@RequestMapping("/")
public class TestController {
@PostMapping("test3")
public User test3(@RequestBody @Validated({ FirstGroup.class }) User u) {
System.out.println(u);
return u;
}
@PostMapping("test4")
public User test4(@Validated User u) {
System.out.println(u);
return u;
}
}
校驗(yàn)失敗,會(huì)直接拋出異常。這樣不太友好,可以使用@ControllerAdvice處理全局異常。
六、定義全局異常處理類
import java.util.List;
import org.springframework.validation.BindException;
import org.springframework.validation.BindingResult;
import org.springframework.validation.FieldError;
import org.springframework.web.bind.MethodArgumentNotValidException;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.bind.annotation.RestControllerAdvice;
import lombok.extern.slf4j.Slf4j;
@Slf4j
@RestControllerAdvice
public class GlobalExceptionHandler {
@ExceptionHandler(value = BindException.class)
public JsonResult exceptionHandle(BindException exception) {
BindingResult result = exception.getBindingResult();
StringBuilder errorMsg = new StringBuilder();
List<FieldError> fieldErrors = result.getFieldErrors();
fieldErrors.forEach(error -> {
log.error("field: " + error.getField() + ", msg:" + error.getDefaultMessage());
errorMsg.append(error.getDefaultMessage()).append("!");
});
return JsonResult.fail(errorMsg.toString());
}
@ExceptionHandler(value = MethodArgumentNotValidException.class)
public JsonResult MyExceptionHandle(MethodArgumentNotValidException exception) {
BindingResult result = exception.getBindingResult();
StringBuilder errorMsg = new StringBuilder();
List<FieldError> fieldErrors = result.getFieldErrors();
fieldErrors.forEach(error -> {
log.error("field: " + error.getField() + ", msg:" + error.getDefaultMessage());
errorMsg.append(error.getDefaultMessage()).append("!");
});
return JsonResult.fail(errorMsg.toString());
}
// 處理運(yùn)行時(shí)異常
@ExceptionHandler(RuntimeException.class)
public JsonResult doHandleRuntimeException(RuntimeException e) {
log.error(e.getMessage(), e);
e.printStackTrace();
return JsonResult.fail(e.getMessage());
}
}
另外JsonResult.java用于接口返回統(tǒng)一個(gè)json格式
import com.fasterxml.jackson.annotation.JsonInclude;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;
import lombok.experimental.Accessors;
/**
*@author wang
*@Date 2020-9-14
*
**/
@Data
@Accessors(chain = true)
@NoArgsConstructor
@AllArgsConstructor
@JsonInclude(JsonInclude.Include.NON_NULL)
public class JsonResult<T> {
/** 成功 */
public static final int SUCCESS=200;
/**內(nèi)部服務(wù)器錯(cuò)誤**/
public static final int FAIL=500;
/** 沒有登錄 */
public static final int NOT_LOGIN = 400;
/** 發(fā)生異常 */
public static final int EXCEPTION = 401;
/** 系統(tǒng)錯(cuò)誤 */
public static final int SYS_ERROR = 402;
/** 參數(shù)錯(cuò)誤 */
public static final int PARAMS_ERROR = 403;
/** 不支持或已經(jīng)廢棄 */
public static final int NOT_SUPPORTED = 410;
/** AuthCode錯(cuò)誤 */
public static final int INVALID_AUTHCODE = 444;
/** 太頻繁的調(diào)用 */
public static final int TOO_FREQUENT = 445;
/** 未知的錯(cuò)誤 */
public static final int UNKNOWN_ERROR = 499;
private Integer code;
private String msg;
private T data;
public static JsonResult fail() {
return new JsonResult(FAIL, "請(qǐng)求處理失敗",null);
}
public static JsonResult fail(String msg) {
return new JsonResult(FAIL, msg,null);
}
public static JsonResult fail(Integer code,String msg) {
return new JsonResult(code, msg,null);
}
public static JsonResult success() {
return new JsonResult(SUCCESS,"請(qǐng)求處理成功",null);
}
public static JsonResult success(String msg) {
return new JsonResult(SUCCESS,msg,null);
}
public static <T> JsonResult success(T data) {
return new JsonResult<T> (SUCCESS,"請(qǐng)求處理成功",data);
}
public static <T> JsonResult success(String msg,T data) {
return new JsonResult<T>(SUCCESS, msg,data);
}
public static JsonResult err() {
return build(EXCEPTION);
}
public static JsonResult err(String msg) {
return build(EXCEPTION, msg);
}
public JsonResult<T> code(int code) {
this.code = code;
return this;
}
public JsonResult<T> msg(String msg) {
this.msg = msg;
return this;
}
public JsonResult<T> data(T data) {
this.data = data;
return this;
}
public static JsonResult build() {
return new JsonResult();
}
public static JsonResult build(int code) {
return new JsonResult().code(code);
}
public static JsonResult build(int code, String msg) {
return new JsonResult<String>().code(code).msg(msg);
}
public static <T> JsonResult<T> build(int code, T data) {
return new JsonResult<T>().code(code).data(data);
}
public static <T> JsonResult<T> build(int code, String msg, T data) {
return new JsonResult<T>().code(code).msg(msg).data(data);
}
}
當(dāng)然還有其他異常處理方式可以參考:http://www.dhdzp.com/article/244379.htm
七、測(cè)試效果



八、嵌套對(duì)象的校驗(yàn)
import lombok.Data;
import javax.validation.Valid;
import javax.validation.constraints.*;
@Data
public class User {
@NotEmpty(message = "用戶名不能為空")
@Size(message = "長(zhǎng)度 [1-3] ", min = 1, max = 3, groups = ValidateGroup.FirstGroup.class)
private String username;
@Min(message = "id不得小于0", value = 0)
private Integer id;
@NotBlank(message = "地址不能為空", groups = {ValidateGroup.ThirdGroup.class, ValidateGroup.SecondeGroup.class})
private String address;
//在內(nèi)部屬性是自定義對(duì)象的時(shí)候添加 @Valid 注解 ,即可開啟對(duì)In對(duì)象的校驗(yàn)。
//記得添加@NotNull注解,否則該對(duì)象可以為null,并且此時(shí)In對(duì)象的校驗(yàn)規(guī)則也不會(huì)拋出異常 。
@Valid
@NotNull(message = "In 對(duì)象不能為null ", groups = {ValidateGroup.ThirdGroup.class, ValidateGroup.SecondeGroup.class})
private In in;
}
/**
*自定義的對(duì)象,User對(duì)象中的一個(gè)屬性。
*/
@Data
public class In {
@NotBlank(message = "str不能為空", groups = {ValidateGroup.ThirdGroup.class, ValidateGroup.SecondeGroup.class})
private String str;
}
九、自定義注解(自定義校驗(yàn)規(guī)則)
9.1 實(shí)現(xiàn)ConstraintValidator接口
import javax.validation.ConstraintValidator;
import javax.validation.ConstraintValidatorContext;
public class WordConstraintValidator implements ConstraintValidator<CustomValidaor, Object> {
@Override
public boolean isValid(Object value, ConstraintValidatorContext context) {
// 具體的校驗(yàn)規(guī)則
return value.toString().length() == 10;
}
}
9.2 自定義注解
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
import javax.validation.Constraint;
import javax.validation.Payload;
@Target({ ElementType.METHOD, ElementType.FIELD })
@Retention(RetentionPolicy.RUNTIME)
@Constraint(validatedBy = WordConstraintValidator.class)
public @interface CustomValidaor {
String message();
// groups 和 payload 這兩個(gè)parameter 必須包含,不然會(huì)報(bào)錯(cuò)
Class<?>[] groups() default {};
Class<? extends Payload>[] payload() default {};
}
9.3 使用
@Data
public class Inner {
@NotBlank(message = "str不能為空", groups = { ValidateGroup.ThirdGroup.class, ValidateGroup.SecondeGroup.class })
@CustomValidaor(message = "長(zhǎng)度必須為10", groups = { ValidateGroup.ThirdGroup.class, ValidateGroup.SecondeGroup.class })
private String str;
}
9.4測(cè)試
訪問接口
@PostMapping("test6")
public User test6(@Validated({ValidateGroup.SecondeGroup.class}) @RequestBody User u) {
System.out.println(u);
return u;
}
到此這篇關(guān)于spring-boot-starter-validation 校驗(yàn)參數(shù)的實(shí)現(xiàn)的文章就介紹到這了,更多相關(guān)spring-boot-starter-validation 校驗(yàn)參數(shù)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- SpringBoot + validation 接口參數(shù)校驗(yàn)的思路詳解
- Spring?Boot集成validation實(shí)現(xiàn)參數(shù)校驗(yàn)功能
- SpringBoot使用Validation包進(jìn)行輸入?yún)?shù)校驗(yàn)
- spring?boot?validation參數(shù)校驗(yàn)與分組嵌套各種類型及使用小結(jié)
- springboot中使用Hibernate-Validation校驗(yàn)參數(shù)詳解
- SpringBoot使用Validation進(jìn)行參數(shù)校驗(yàn)的示例詳解
- spring boot輸入數(shù)據(jù)校驗(yàn)(validation)的實(shí)現(xiàn)過程
- 從零到掌握Spring Boot Validation 接口校驗(yàn)的詳細(xì)過程
相關(guān)文章
Mapper類中存在名稱相同的方法重載報(bào)錯(cuò)問題
這篇文章主要介紹了Mapper類中存在名稱相同的方法重載報(bào)錯(cuò)問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-03-03
深入分析Spring Cloud 負(fù)載均衡器架構(gòu)選型
文章詳細(xì)介紹了OpenFeign的初始化和執(zhí)行流程,包括FeignClient注解、FeignClientsRegistrar類、FeignClientFactoryBean工廠類、Targeter接口及其實(shí)現(xiàn)等,感興趣的朋友一起看看吧2024-12-12
lambda表達(dá)式與傳統(tǒng)接口函數(shù)實(shí)現(xiàn)方式對(duì)比詳解
這篇文章主要為大家介紹了lambda表達(dá)式與傳統(tǒng)接口函數(shù)實(shí)現(xiàn)方式對(duì)比詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家度偶多進(jìn)步早日升職加薪2022-03-03
Java System.getProperty()-獲取系統(tǒng)參數(shù)案例詳解
這篇文章主要介紹了Java System.getProperty()-獲取系統(tǒng)參數(shù)案例詳解,本篇文章通過簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-08-08
MyBatis中一對(duì)多的xml配置方式(嵌套查詢/嵌套結(jié)果)
這篇文章主要介紹了MyBatis中一對(duì)多的xml配置方式(嵌套查詢/嵌套結(jié)果),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-03-03

