国产激情自拍_国产9色视频_丁香花在线电影小说观看 _久久久久国产精品嫩草影院

首頁 > 開發 > Java > 正文

詳解使用spring validation完成數據后端校驗

2024-07-14 08:43:51
字體:
來源:轉載
供稿:網友

前言

數據的校驗是交互式網站一個不可或缺的功能,前端的js校驗可以涵蓋大部分的校驗職責,如用戶名唯一性,生日格式,郵箱格式校驗等等常用的校驗。但是為了避免用戶繞過瀏覽器,使用http工具直接向后端請求一些違法數據,服務端的數據校驗也是必要的,可以防止臟數據落到數據庫中,如果數據庫中出現一個非法的郵箱格式,也會讓運維人員頭疼不已。我在之前保險產品研發過程中,系統對數據校驗要求比較嚴格且追求可變性及效率,曾使用drools作為規則引擎,兼任了校驗的功能。而在一般的應用,可以使用本文將要介紹的validation來對數據進行校驗。

簡述JSR303/JSR-349,hibernate validation,spring validation之間的關系。JSR303是一項標準,JSR-349是其的升級版本,添加了一些新特性,他們規定一些校驗規范即校驗注解,如@Null,@NotNull,@Pattern,他們位于javax.validation.constraints包下,只提供規范不提供實現。而hibernate validation是對這個規范的實踐(不要將hibernate和數據庫orm框架聯系在一起),他提供了相應的實現,并增加了一些其他校驗注解,如@Email,@Length,@Range等等,他們位于org.hibernate.validator.constraints包下。而萬能的spring為了給開發者提供便捷,對hibernate validation進行了二次封裝,顯示校驗validated bean時,你可以使用spring validation或者hibernate validation,而spring validation另一個特性,便是其在springmvc模塊中添加了自動校驗,并將校驗信息封裝進了特定的類中。這無疑便捷了我們的web開發。本文主要介紹在springmvc中自動校驗的機制。

引入依賴

我們使用maven構建springboot應用來進行demo演示。

<dependencies>  <dependency>    <groupId>org.springframework.boot</groupId>    <artifactId>spring-boot-starter-web</artifactId>  </dependency></dependencies>

我們只需要引入spring-boot-starter-web依賴即可,如果查看其子依賴,可以發現如下的依賴:

<dependency>  <groupId>org.hibernate</groupId>  <artifactId>hibernate-validator</artifactId></dependency><dependency>  <groupId>com.fasterxml.jackson.core</groupId>  <artifactId>jackson-databind</artifactId></dependency>

驗證了我之前的描述,web模塊使用了hibernate-validation,并且databind模塊也提供了相應的數據綁定功能。

構建啟動類

無需添加其他注解,一個典型的啟動類

@SpringBootApplicationpublic class ValidateApp {  public static void main(String[] args) {    SpringApplication.run(ValidateApp.class, args);  }}

創建需要被校驗的實體類

public class Foo {  @NotBlank  private String name;  @Min(18)  private Integer age;  @Pattern(regexp = "^1(3|4|5|7|8)//d{9}$",message = "手機號碼格式錯誤")  @NotBlank(message = "手機號碼不能為空")  private String phone;  @Email(message = "郵箱格式錯誤")  private String email;  //... getter setter}

使用一些比較常用的校驗注解,還是比較淺顯易懂的,字段上的注解名稱即可推斷出校驗內容,每一個注解都包含了message字段,用于校驗失敗時作為提示信息,特殊的校驗注解,如Pattern(正則校驗),還可以自己添加正則表達式

在@Controller中校驗數據

springmvc為我們提供了自動封裝表單參數的功能,一個添加了參數校驗的典型controller如下所示。

@Controllerpublic class FooController {  @RequestMapping("/foo")  public String foo(@Validated Foo foo <1>, BindingResult bindingResult <2>) {    if(bindingResult.hasErrors()){      for (FieldError fieldError : bindingResult.getFieldErrors()) {        //...      }      return "fail";    }    return "success";  }}

值得注意的地方:

<1> 參數Foo前需要加上@Validated注解,表明需要spring對其進行校驗,而校驗的信息會存放到其后的BindingResult中。注意,必須相鄰,如果有多個參數需要校驗,形式可以如下。foo(@Validated Foo foo, BindingResult fooBindingResult ,@Validated Bar bar, BindingResult barBindingResult);即一個校驗類對應一個校驗結果。

<2> 校驗結果會被自動填充,在controller中可以根據業務邏輯來決定具體的操作,如跳轉到錯誤頁面。

一個最基本的校驗就完成了,總結下框架已經提供了哪些校驗:

JSR提供的校驗注解: 
@Null 被注釋的元素必須為 null 
@NotNull 被注釋的元素必須不為 null 
@AssertTrue 被注釋的元素必須為 true 
@AssertFalse 被注釋的元素必須為 false 
@Min(value) 被注釋的元素必須是一個數字,其值必須大于等于指定的最小值 
@Max(value) 被注釋的元素必須是一個數字,其值必須小于等于指定的最大值 
@DecimalMin(value) 被注釋的元素必須是一個數字,其值必須大于等于指定的最小值 
@DecimalMax(value) 被注釋的元素必須是一個數字,其值必須小于等于指定的最大值 
@Size(max=, min=) 被注釋的元素的大小必須在指定的范圍內 
@Digits (integer, fraction) 被注釋的元素必須是一個數字,其值必須在可接受的范圍內 
@Past 被注釋的元素必須是一個過去的日期 
@Future 被注釋的元素必須是一個將來的日期 
@Pattern(regex=,flag=) 被注釋的元素必須符合指定的正則表達式

Hibernate Validator提供的校驗注解: 
@NotBlank(message =) 驗證字符串非null,且長度必須大于0 
@Email 被注釋的元素必須是電子郵箱地址 
@Length(min=,max=) 被注釋的字符串的大小必須在指定的范圍內 
@NotEmpty 被注釋的字符串的必須非空 
@Range(min=,max=,message=) 被注釋的元素必須在合適的范圍內

校驗實驗

我們對上面實現的校驗入口進行一次測試請求: 

訪問 http://localhost:8080/foo?name=xujingfeng&email=000&age=19 可以得到如下的debug信息:

spring,validation,數據后端校驗

實驗告訴我們,校驗結果起了作用。并且,可以發現當發生多個錯誤,spring validation不會在第一個錯誤發生后立即停止,而是繼續試錯,告訴我們所有的錯誤。debug可以查看到更多豐富的錯誤信息,這些都是spring validation為我們提供的便捷特性,基本適用于大多數場景。

你可能不滿足于簡單的校驗特性,下面進行一些補充。

分組校驗

如果同一個類,在不同的使用場景下有不同的校驗規則,那么可以使用分組校驗。未成年人是不能喝酒的,而在其他場景下我們不做特殊的限制,這個需求如何體現同一個實體,不同的校驗規則呢?

改寫注解,添加分組:

Class Foo{  @Min(value = 18,groups = {Adult.class})  private Integer age;  public interface Adult{}  public interface Minor{}}

這樣表明,只有在Adult分組下,18歲的限制才會起作用。

Controller層改寫:

@RequestMapping("/drink")public String drink(@Validated({Foo.Adult.class}) Foo foo, BindingResult bindingResult) {  if(bindingResult.hasErrors()){    for (FieldError fieldError : bindingResult.getFieldErrors()) {      //...    }    return "fail";  }  return "success";}@RequestMapping("/live")public String live(@Validated Foo foo, BindingResult bindingResult) {  if(bindingResult.hasErrors()){    for (FieldError fieldError : bindingResult.getFieldErrors()) {      //...    }    return "fail";  }  return "success";}

drink方法限定需要進行Adult校驗,而live方法則不做限制。

自定義校驗

業務需求總是比框架提供的這些簡單校驗要復雜的多,我們可以自定義校驗來滿足我們的需求。自定義spring validation非常簡單,主要分為兩步。

1 自定義校驗注解 

我們嘗試添加一個“字符串不能包含空格”的限制。

@Target({METHOD, FIELD, ANNOTATION_TYPE, CONSTRUCTOR, PARAMETER})@Retention(RUNTIME)@Documented@Constraint(validatedBy = {CannotHaveBlankValidator.class})<1>public @interface CannotHaveBlank {  //默認錯誤消息  String message() default "不能包含空格";  //分組  Class<?>[] groups() default {};  //負載  Class<? extends Payload>[] payload() default {};  //指定多個時使用  @Target({FIELD, METHOD, PARAMETER, ANNOTATION_TYPE})  @Retention(RUNTIME)  @Documented  @interface List {    CannotHaveBlank[] value();  }}

我們不需要關注太多東西,使用spring validation的原則便是便捷我們的開發,例如payload,List ,groups,都可以忽略。

<1> 自定義注解中指定了這個注解真正的驗證者類。

2 編寫真正的校驗者類

public class CannotHaveBlankValidator implements <1> ConstraintValidator<CannotHaveBlank, String> {  @Override  public void initialize(CannotHaveBlank constraintAnnotation) {  }  @Override  public boolean isValid(String value, ConstraintValidatorContext context <2>) {    //null時不進行校驗    if (value != null && value.contains(" ")) {      <3>      //獲取默認提示信息      String defaultConstraintMessageTemplate = context.getDefaultConstraintMessageTemplate();      System.out.println("default message :" + defaultConstraintMessageTemplate);      //禁用默認提示信息      context.disableDefaultConstraintViolation();      //設置提示語      context.buildConstraintViolationWithTemplate("can not contains blank").addConstraintViolation();      return false;    }    return true;  }}

<1> 所有的驗證者都需要實現ConstraintValidator接口,它的接口也很形象,包含一個初始化事件方法,和一個判斷是否合法的方法。

public interface ConstraintValidator<A extends Annotation, T> {  void initialize(A constraintAnnotation);  boolean isValid(T value, ConstraintValidatorContext context);}

<2> ConstraintValidatorContext 這個上下文包含了認證中所有的信息,我們可以利用這個上下文實現獲取默認錯誤提示信息,禁用錯誤提示信息,改寫錯誤提示信息等操作。

<3> 一些典型校驗操作,或許可以對你產生啟示作用。

值得注意的一點是,自定義注解可以用在METHOD, FIELD, ANNOTATION_TYPE, CONSTRUCTOR, PARAMETER之上,ConstraintValidator的第二個泛型參數T,是需要被校驗的類型。

手動校驗

可能在某些場景下需要我們手動校驗,即使用校驗器對需要被校驗的實體發起validate,同步獲得校驗結果。理論上我們既可以使用Hibernate Validation提供Validator,也可以使用Spring對其的封裝。在spring構建的項目中,提倡使用經過spring封裝過后的方法,這里兩種方法都介紹下:

Hibernate Validation:

Foo foo = new Foo();foo.setAge(22);foo.setEmail("000");ValidatorFactory vf = Validation.buildDefaultValidatorFactory();Validator validator = vf.getValidator();Set<ConstraintViolation<Foo>> set = validator.validate(foo);for (ConstraintViolation<Foo> constraintViolation : set) {  System.out.println(constraintViolation.getMessage());}

由于依賴了Hibernate Validation框架,我們需要調用Hibernate相關的工廠方法來獲取validator實例,從而校驗。

在spring framework文檔的Validation相關章節,可以看到如下的描述:

Spring provides full support for the Bean Validation API. This includes convenient support for bootstrapping a JSR-303/JSR-349 Bean Validation provider as a Spring bean. This allows for a javax.validation.ValidatorFactory or javax.validation.Validator to be injected wherever validation is needed in your application. Use the LocalValidatorFactoryBean to configure a default Validator as a Spring bean:

bean id=”validator” class=”org.springframework.validation.beanvalidation.LocalValidatorFactoryBean”

The basic configuration above will trigger Bean Validation to initialize using its default bootstrap mechanism. A JSR-303/JSR-349 provider, such as Hibernate Validator, is expected to be present in the classpath and will be detected automatically.

上面這段話主要描述了spring對validation全面支持JSR-303、JSR-349的標準,并且封裝了LocalValidatorFactoryBean作為validator的實現。值得一提的是,這個類的責任其實是非常重大的,他兼容了spring的validation體系和hibernate的validation體系,也可以被開發者直接調用,代替上述的從工廠方法中獲取的hibernate validator。由于我們使用了springboot,會觸發web模塊的自動配置,LocalValidatorFactoryBean已經成為了Validator的默認實現,使用時只需要自動注入即可。

@AutowiredValidator globalValidator; <1>@RequestMapping("/validate")public String validate() {  Foo foo = new Foo();  foo.setAge(22);  foo.setEmail("000");  Set<ConstraintViolation<Foo>> set = globalValidator.validate(foo);<2>  for (ConstraintViolation<Foo> constraintViolation : set) {    System.out.println(constraintViolation.getMessage());  }  return "success";}

<1> 真正使用過Validator接口的讀者會發現有兩個接口,一個是位于javax.validation包下,另一個位于org.springframework.validation包下,注意我們這里使用的是前者javax.validation,后者是spring自己內置的校驗接口,LocalValidatorFactoryBean同時實現了這兩個接口。

<2> 此處校驗接口最終的實現類便是LocalValidatorFactoryBean。

基于方法校驗

@RestController@Validated <1>public class BarController {  @RequestMapping("/bar")  public @NotBlank <2> String bar(@Min(18) Integer age <3>) {    System.out.println("age : " + age);    return "";  }  @ExceptionHandler(ConstraintViolationException.class)  public Map handleConstraintViolationException(ConstraintViolationException cve){    Set<ConstraintViolation<?>> cves = cve.getConstraintViolations();<4>    for (ConstraintViolation<?> constraintViolation : cves) {      System.out.println(constraintViolation.getMessage());    }    Map map = new HashMap();    map.put("errorCode",500);    return map;  }}

<1> 為類添加@Validated注解

<2> <3> 校驗方法的返回值和入參

<4> 添加一個異常處理器,可以獲得沒有通過校驗的屬性相關信息

基于方法的校驗,個人不推薦使用,感覺和項目結合的不是很好。

使用校驗框架的一些想法

理論上spring validation可以實現很多復雜的校驗,你甚至可以使你的Validator獲取ApplicationContext,獲取spring容器中所有的資源,進行諸如數據庫校驗,注入其他校驗工具,完成組合校驗(如前后密碼一致)等等操作,但是尋求一個易用性和封裝復雜性之間的平衡點是我們作為工具使用者應該考慮的,我推崇的方式,是僅僅使用自帶的注解和自定義注解,完成一些簡單的,可復用的校驗。而對于復雜的校驗,則包含在業務代碼之中,畢竟如用戶名是否存在這樣的校驗,僅僅依靠數據庫查詢還不夠,為了避免并發問題,還是得加上唯一索引之類的額外工作,不是嗎?

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持VeVb武林網。


注:相關教程知識閱讀請移步到JAVA教程頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
国产激情自拍_国产9色视频_丁香花在线电影小说观看 _久久久久国产精品嫩草影院
成网站在线观看人免费| 国产成+人+亚洲+欧美+综合| 国产精品亚洲色图| av在线播放网| 人成在线免费视频| 久艹在线视频| 国产日韩欧美第一页| 国产乱子视频| 国产精品日日爱| 精品一区二区三区高清免费不卡| 国产成人亚洲综合小说区 | 日本一二区视频| 国产午夜精品一区理论片| 国产精品视频二区三区| 国产黄在线播放| 亚洲视频在线观看不卡| 成年网站免费入口在线观看| 日本最新在线视频| 在线国产一区二区三区| 欧美日韩国产亚洲沙发| 国产一级黄色| yjizz视频网站在线播放| 成人免费一区二区三区牛牛 | 在线激情网站| 国产精品一品| 亚洲精品国自产拍在线观看| 国产裸舞福利在线视频合集| 国产黄网站在线观看| 狠狠狠狠狠狠操| 亚洲精品午夜级久久久久| 国产九色视频| 亚洲第一页在线播放| 国产高清在线a视频大全| 开心激情五月婷婷| av中文网站| 9色在线视频网站| 国产三区视频在线观看| 日本卡一卡2卡3卡4精品卡网站| 国产一级影片| 天堂在线一二区| 欧美日韩国产亚洲沙发| 在线黄色国产电影| 亚洲欧美自拍另类| 亚洲国产成人综合| 麻豆福利在线观看| 精品资源在线看| 欧美成人亚洲高清在线观看| av人人综合网| 天天干天天摸| 青青青手机在线视频观看| 在线国产福利网站| 亚洲成人福利| av高清在线| 在线成人综合色一区| 在线中文字幕av| 在线国产小视频| www亚洲天堂| 免费观看久久久久| 国产丝袜精品丝袜| 国产极品嫩模在线视频一区| 91最新在线| 操操操综合网| 91xxx在线观看| 国产精品69一区二区三区| 国产精品伦一区二区三区级视频频| 中文字幕有码在线视频| 尤物视频在线观看| 尤物免费看在线视频| 国产色在线播放| 四虎www视频| 国产69久久| 18av在线播放| 久久精品国产亚洲a∨麻豆| 国产鲁鲁视频在线观看特色| 国产二区在线播放| 黄网址在线永久免费观看| 国产精选在线观看| 在线伊人免费视频| 国产中文字幕在线| 伊人春色在线| 国产丝袜精品丝袜| 久久国产情侣| 91午夜视频| 中文字幕在线免费观看| 国产99re66在线视频| 亚洲精品国自产拍在线观看| 六月天色婷婷| 在线观看精品一区二区三区| 国产乱子伦三级在线播放| 精精国产xxxx视频在线中文版| 超碰在线网址| аⅴ成人天堂中文在线| 精品美女在线观看视频在线观看 | 2021av在线| 国产卡二和卡三的视频| 免费a级毛片在线观看| 精品成人一区二区三区免费视频| а√天堂www在线а√天堂视频| 国产特级淫片免费看| 免费不卡中文字幕视频| 在线看黄网址| av在线免费播放| www.色婷婷| 激情综合丁香| 激情网站在线| 豆国产97在线|亚洲| 日本高清中文字幕在线 | av麻豆国产| 在线午夜视频| 一个人看的www免费观看视频| 91蜜桃在线视频| 91黄色在线| eeuss影院在线观看第一页| 精品一区二区三区免费站| 精品视频在线一区二区| 国产国语**毛片高清视频 | 亚洲视频精品在线观看| 天天操夜夜做| 国产免费高清| 国产一卡2卡3卡免费网站| 国产激情99| 香蕉视频在线看| 国产中文字幕在线观看| 国产粉嫩一区二区三区在线观看| 伊人永久在线| 国产鲁鲁视频在线观看特色| 国产精品ⅴa有声小说| 亚洲91av| 最近久乱中文字幕| 亚洲天堂影院在线观看| 四虎影院成人| 国产欧美日韩专区| 免费高清视频日韩| 国产激情网址| 国产麻豆免费| 欧美亚洲另类在线观看| 五月婷婷在线观看| 日本在线天堂| 国产视频中文字幕| 蜜桃视频中文字幕| 中文字幕av在线播放| 成人精品一区二区三区免费| 国产国产国产国产国产国产| 九九免费视频| 男人天堂v视频| 91中文字幕| 国产小视频在线| 好男人免费精品视频| 国产精品免费麻豆入口| 国产在线一二| 免费午夜一级| а√最新版在线天堂| 国产成人精品实拍在线| 国产精品亚洲第五区在线| 青青久草在线| www.麻豆av.com| 日韩精品免费一区二区| 天天噜天天色| а√最新版地址在线天堂| 在线观看免费高清完整| 最新国产在线| 中文字幕在线免费观看| 中文字幕亚洲精品视频| 五月婷婷在线观看| 国产一区二区三区不卡免费观看 | 国产youjizz在线| 国产精品久久久久久久久鸭 | eeuss影院在线播放| 亚洲国产日韩在线人成电影| 国产精品69一区二区三区| 中文字幕成人乱码在线电影| 国产精品一区二区三区四区色| 黄网址在线永久免费观看| 福利在线视频导航| 国产一级影片| 一个人看的www免费观看视频| 在线一区观看| а√最新版在线天堂| 天天操人人干| 中文字幕2020第一页| 免费久久网站| 国产视频资源| 亚洲欧美久久婷婷爱综合一区天堂| 欧美性受xxxx免费视频| 超碰在线网站| 2018中文字幕在线| 啪啪免费视频一区| 国产视频青青| av在线网页| 国产网站免费观看| a视频免费看| 国产视频福利| 蜜桃av网站| 在线黄色国产电影| 国产一级二级三级在线观看| 亚洲日本一区二区三区在线观看| 久久精品视频观看| 97国产视频| 国产天堂av| 久热久精久品这里在线观看|