Spring?Boot集成validation實現(xiàn)參數(shù)校驗功能
1.什么是Bean Validation?
Bean Validation 是一個運行時的數(shù)據(jù)驗證框架,在驗證之后驗證的錯誤信息會被馬上返回。java 在2009年的 JAVAEE 6 中發(fā)布了 JSR303以及javax下的validation包內(nèi)容。這項工作的主要目標是為java應(yīng)用程序開發(fā)人員提供 基于java對象的 約束(constraints)聲明 和 對約束的驗證工具(validator),以及約束元數(shù)據(jù)存儲庫和查詢API。但是該內(nèi)容并沒有具體的實現(xiàn), Hibernate-Validator框架 提供了 JSR 303 規(guī)范中所有內(nèi)置 constraint 的實現(xiàn),除此之外還有一些附加的 constraint。
常用的Validation注解
一些最常見的驗證注解如下:
@NotNull
: 標記字段不能為null
@NotEmpty
: 標記集合字段不為空(至少要有一個元素)@NotBlank
: 標記字段串字段不能是空字符串(即它必須至少有一個字符)@Min
/@Max
: 標記數(shù)字類型字段必須大于/小于指定的值@Pattern
: 標記字符串字段必須匹配指定的正則表達式@Email
: 標記字符串字段必須是有效的電子郵件地址
2.代碼工程
實驗?zāi)繕耍盒r瀋ontroller傳遞的參數(shù)
pom.xml
<?xml version="1.0" encoding="UTF-8"?> <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> <parent> <artifactId>springboot-demo</artifactId> <groupId>com.et</groupId> <version>1.0-SNAPSHOT</version> </parent> <modelVersion>4.0.0</modelVersion> <artifactId>validtion</artifactId> <properties> <maven.compiler.source>8</maven.compiler.source> <maven.compiler.target>8</maven.compiler.target> </properties> <dependencies> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-autoconfigure</artifactId> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-test</artifactId> <scope>test</scope> </dependency> <dependency> <groupId>org.projectlombok</groupId> <artifactId>lombok</artifactId> </dependency> <dependency> <groupId>cn.hutool</groupId> <artifactId>hutool-all</artifactId> <version>5.8.16</version> </dependency> </dependencies> </project>
controller
Spring 自動將傳入的 JSON 映射到 Java 對象參數(shù)上。 現(xiàn)在,我們要校驗傳入的 Java 對象是否滿足我們預(yù)先定義的約束條件。 為了校驗傳入 HTTP 請求的請求實體,我們在 REST 控制器中使用 @Valid
注解對請求實體進行標記:
package com.et.validation.controller; import cn.hutool.json.JSON; import cn.hutool.json.JSONObject; import cn.hutool.json.JSONUtil; import com.et.validation.entity.UserInfoReq; import com.fasterxml.jackson.databind.util.JSONPObject; import org.springframework.validation.annotation.Validated; import org.springframework.web.bind.annotation.RequestBody; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.RestController; import java.util.HashMap; import java.util.Map; @RestController public class HelloWorldController { @RequestMapping("/hello") public Map<String, Object> showHelloWorld(@RequestBody @Validated UserInfoReq req){ Map<String, Object> map = new HashMap<>(); map.put("msg", JSONUtil.toJsonStr(req)); return map; } }
enttiy
我們有一個 int類型 字段,它的值必須介于 1 和 10200之間,如@Min
和@Max
注解所定義的那樣。 我們還有一個 String 類型字段,它必須是一個 IP 地址,正如@Pattern
注解中的正則表達式所定義的那樣(正則表達式實際上仍然允許大于 255 的無效 IP 地址,但在我們創(chuàng)建自定義驗證器時,我們將在本教程的后面修復這個BUG)。
package com.et.validation.entity; import com.et.validation.validate.IpAddress; import lombok.Data; import lombok.ToString; import javax.validation.constraints.*; @Data @ToString public class UserInfoReq { @NotNull(message = "id不能為null") private Long id; @NotBlank(message = "username不能為空") private String username; @NotNull(message = "age不能為null") @Min(value = 1, message = "年齡不符合要求") @Max(value = 200, message = "年齡不符合要求") private Integer age; @Email(message = "郵箱不符合規(guī)范") private String email; @IpAddress(message = "ip不符合規(guī)范") //@Pattern(regexp = "^[0-9]{1,3}\\.[0-9]{1,3}\\.[0-9]{1,3}\\.[0-9]{1,3}$") private String ip; }
自定義error捕獲
當校驗失敗時,我們通常希望向客戶端返回一條有意義的錯誤消息。 為了使客戶端能夠顯示有用的錯誤消息,我們應(yīng)該返回一個統(tǒng)一的數(shù)據(jù)結(jié)構(gòu),其中包含每個校驗失敗的錯誤消息。 我們在這里所做的只是從異常中讀取有關(guān)校驗失敗信息并將它們轉(zhuǎn)換到我們的 ValidationErrorResponse
數(shù)據(jù)結(jié)構(gòu)中。 請注意@ControllerAdvice
注解,它使得上述類型異常的異常處理機制對所有Controller全局可用。
package com.et.validation.error; import org.springframework.http.HttpStatus; import org.springframework.validation.FieldError; import org.springframework.web.bind.MethodArgumentNotValidException; import org.springframework.web.bind.annotation.ControllerAdvice; import org.springframework.web.bind.annotation.ExceptionHandler; import org.springframework.web.bind.annotation.ResponseBody; import org.springframework.web.bind.annotation.ResponseStatus; import javax.validation.ConstraintViolation; import javax.validation.ConstraintViolationException; @ControllerAdvice class ErrorHandlingControllerAdvice { @ExceptionHandler(ConstraintViolationException.class) @ResponseStatus(HttpStatus.BAD_REQUEST) @ResponseBody ValidationErrorResponse onConstraintValidationException( ConstraintViolationException e) { ValidationErrorResponse error = new ValidationErrorResponse(); for (ConstraintViolation violation : e.getConstraintViolations()) { error.getViolations().add( new Violation(violation.getPropertyPath().toString(), violation.getMessage())); } return error; } @ExceptionHandler(MethodArgumentNotValidException.class) @ResponseStatus(HttpStatus.BAD_REQUEST) @ResponseBody ValidationErrorResponse onMethodArgumentNotValidException( MethodArgumentNotValidException e) { ValidationErrorResponse error = new ValidationErrorResponse(); for (FieldError fieldError : e.getBindingResult().getFieldErrors()) { error.getViolations().add( new Violation(fieldError.getField(), fieldError.getDefaultMessage())); } return error; } }
package com.et.validation.error; import lombok.Data; import java.io.Serializable; import java.util.ArrayList; import java.util.List; @Data public class ValidationErrorResponse implements Serializable { private List<Violation> violations = new ArrayList<>(); }
package com.et.validation.error; import lombok.Data; import java.io.Serializable; @Data public class Violation implements Serializable { private final String fieldName; private final String message; public Violation(String fieldName, String message) { this.fieldName = fieldName; this.message = message; } }
自定義validator
如果官方提供可用的校驗注解不能滿足我們的需要,我們自己可以定義一個自定義校驗器。 自定義校驗注解需要包含以下要素:
- 參數(shù)
message
, 指定 ValidationMessages.properties 文件中的屬性鍵,用于在校驗失敗時解析提示消息, - 參數(shù)
groups
, 允許定義在何種情況下觸發(fā)此校驗(稍后我們將討分組校驗) - 參數(shù)
payload
, 允許定義要通過此校驗傳遞的Payload(因為這是一個很少使用的功能,我們不會在本教程中介紹它) - 一個
@Constraint
注解, 指定實現(xiàn) ConstraintValidator 接口的校驗邏輯類。
校驗器的實現(xiàn)如下所示:
package com.et.validation.validate; import javax.validation.Constraint; import javax.validation.Payload; import java.lang.annotation.Documented; import java.lang.annotation.Retention; import java.lang.annotation.Target; import static java.lang.annotation.ElementType.FIELD; import static java.lang.annotation.RetentionPolicy.RUNTIME; @Target({ FIELD }) @Retention(RUNTIME) @Constraint(validatedBy = IpAddressValidator.class) @Documented public @interface IpAddress { String message() default "{IpAddress.invalid}"; Class<?>[] groups() default { }; Class<? extends Payload>[] payload() default { }; }
package com.et.validation.validate; import javax.validation.ConstraintValidator; import javax.validation.ConstraintValidatorContext; import java.util.regex.Matcher; import java.util.regex.Pattern; class IpAddressValidator implements ConstraintValidator<IpAddress, String> { @Override public boolean isValid(String value, ConstraintValidatorContext context) { Pattern pattern = Pattern.compile("^([0-9]{1,3})\\.([0-9]{1,3})\\.([0-9]{1,3})\\.([0-9]{1,3})$"); Matcher matcher = pattern.matcher(value); try { if (!matcher.matches()) { return false; } else { for (int i = 1; i <= 4; i++) { int octet = Integer.valueOf(matcher.group(i)); if (octet > 255) { return false; } } return true; } } catch (Exception e) { return false; } } }
以上只是一些關(guān)鍵代碼,所有代碼請參見下面代碼倉庫
代碼倉庫
3.測試
啟動spring boot應(yīng)用
測試參數(shù)校驗接口
訪問http://127.0.0.1:8088/hello,返回結(jié)果如下
4.引用
到此這篇關(guān)于Spring Boot集成validation實現(xiàn)參數(shù)校驗功能的文章就介紹到這了,更多相關(guān)Spring Boot集成validation參數(shù)校驗內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
教你用MAT工具分析Java堆內(nèi)存泄漏問題的解決方法
今天給大家?guī)淼氖顷P(guān)于Java的相關(guān)知識,文章圍繞著如何使用MAT工具分析Java堆內(nèi)存泄漏問題的解決方法展開,文中有非常詳細的介紹及代碼示例,需要的朋友可以參考下2021-06-06兩個小例子輕松搞懂 java 中遞歸與尾遞歸的優(yōu)化操作
這篇文章主要介紹了兩個小例子輕松搞懂 java 中遞歸與尾遞歸的優(yōu)化操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-09-09intellij IDEA配置springboot的圖文教程
Spring Boot是由Pivotal團隊提供的全新框架,其設(shè)計目的是用來簡化新Spring應(yīng)用的初始搭建以及開發(fā)過程。接下來通過本文給大家介紹intellij IDEA配置springboot的圖文教程,感興趣的朋友一起看看吧2018-03-03java 利用java反射機制動態(tài)加載類的簡單實現(xiàn)
下面小編就為大家?guī)硪黄猨ava 利用java反射機制動態(tài)加載類的簡單實現(xiàn)。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2016-09-09解決kafka:org.apache.kafka.common.errors.TimeoutException問題
這篇文章主要介紹了解決kafka:org.apache.kafka.common.errors.TimeoutException問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-01-01