spring注解之@Valid和@Validated的區(qū)分總結(jié)
@Valid:
@Valid注解用于校驗,所屬包為:javax.validation.Valid。
① 首先需要在實體類的相應(yīng)字段上添加用于充當校驗條件的注解,如:@Min,如下代碼(age屬于Girl類中的屬性):
@Min(value = 18,message = "未成年禁止入內(nèi)") private Integer age;
② 其次在controller層的方法的要校驗的參數(shù)上添加@Valid注解,并且需要傳入BindingResult對象,用于獲取校驗失敗情況下的反饋信息,如下代碼:
@PostMapping("/girls") public Girl addGirl(@Valid Girl girl, BindingResult bindingResult) { if(bindingResult.hasErrors()){ System.out.println(bindingResult.getFieldError().getDefaultMessage()); return null; } return girlResposity.save(girl); }
bindingResult.getFieldError.getDefaultMessage()用于獲取相應(yīng)字段上添加的message中的內(nèi)容,如:@Min注解中message屬性的內(nèi)容
@Validated:
@Valid是javax.validation里的。
@Validated是@Valid 的一次封裝,是Spring提供的校驗機制使用。@Valid不提供分組功能
@Validated的特殊用法
1、分組
當一個實體類需要多種驗證方式時,例:對于一個實體類的id來說,新增的時候是不需要的,對于更新時是必須的。
可以通過groups對驗證進行分組
分組接口類(通過向groups分配不同類的class對象,達到分組目的):
package com.valid.interfaces; public interface First { }
實體類:
package com.valid.pojo; import javax.validation.constraints.Size; import org.hibernate.validator.constraints.NotEmpty; import com.valid.interfaces.First; public class People { //在First分組時,判斷不能為空 @NotEmpty(groups={First.class}) private String id; //name字段不為空,且長度在3-8之間 @NotEmpty @Size(min=3,max=8) private String name; public String getName() { return name; } public void setName(String name) { this.name = name; } public String getId() { return id; } public void setId(String id) { this.id = id; } }
注:
(1)不分配groups,默認每次都要進行驗證
(2)對一個參數(shù)需要多種驗證方式時,也可通過分配不同的組達到目的。例:
@NotEmpty(groups={First.class}) @Size(min=3,max=8,groups={Second.class}) private String name;
控制類:
package com.valid.controller; import org.springframework.stereotype.Controller; import org.springframework.validation.BindingResult; import org.springframework.validation.annotation.Validated; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.ResponseBody; import com.valid.interfaces.First; import com.valid.pojo.People; @Controller public class FirstController { @RequestMapping("/addPeople") //不需驗證ID public @ResponseBody String addPeople(@Validated People p,BindingResult result) { System.out.println("people's ID:" + p.getId()); if(result.hasErrors()) { return "0"; } return "1"; } @RequestMapping("/updatePeople") //需要驗證ID public @ResponseBody String updatePeople(@Validated({First.class}) People p,BindingResult result) { System.out.println("people's ID:" + p.getId()); if(result.hasErrors()) { return "0"; } return "1"; } }
注:
@Validated沒有添加groups屬性時,默認驗證沒有分組的驗證屬性,如該例子:People的name屬性。
@Validated沒有添加groups屬性時,所有參數(shù)的驗證類型都有分組(即本例中People的name的@NotEmpty、@Size都添加groups屬性),則不驗證任何參數(shù)
2、組序列
默認情況下,不同組別的約束驗證是無序的,然而在某些情況下,約束驗證的順序卻很重要。
例:
(1)第二個組中的約束驗證依賴于一個穩(wěn)定狀態(tài)來運行,而這個穩(wěn)定狀態(tài)是由第一個組來進行驗證的。
(2)某個組的驗證比較耗時,CPU 和內(nèi)存的使用率相對比較大,最優(yōu)的選擇是將其放在最后進行驗證。因此,在進行組驗證的時候尚需提供一種有序的驗證方式,這就提出了組序列的概念。
一個組可以定義為其他組的序列,使用它進行驗證的時候必須符合該序列規(guī)定的順序。在使用組序列驗證的時候,如果序列前邊的組驗證失敗,則后面的組將不再給予驗證。
分組接口類 (通過@GroupSequence注解對組進行排序):
package com.valid.interfaces; public interface First { }
package com.valid.interfaces; public interface Second { }
package com.valid.interfaces; import javax.validation.GroupSequence; @GroupSequence({First.class,Second.class}) public interface Group { }
實體類:
package com.valid.pojo; import javax.validation.constraints.Size; import org.hibernate.validator.constraints.NotEmpty; import com.valid.interfaces.First; import com.valid.interfaces.Second; public class People { //在First分組時,判斷不能為空 @NotEmpty(groups={First.class}) private String id; //name字段不為空,且長度在3-8之間 @NotEmpty(groups={First.class}) @Size(min=3,max=8,groups={Second.class}) private String name; public String getName() { return name; } public void setName(String name) { this.name = name; } public String getId() { return id; } public void setId(String id) { this.id = id; } }
控制類:
package com.valid.controller; import org.springframework.stereotype.Controller; import org.springframework.validation.BindingResult; import org.springframework.validation.annotation.Validated; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.ResponseBody; import com.valid.interfaces.Group; import com.valid.pojo.People; import com.valid.pojo.Person; @Controller public class FirstController { @RequestMapping("/addPeople") //不需驗證ID public @ResponseBody String addPeople(@Validated({Group.class}) People p,BindingResult result) { if(result.hasErrors()) { return "0"; } return "1"; } }
3、驗證多個對象
一個功能方法上處理多個模型對象時,需添加多個驗證結(jié)果對象
package com.valid.controller; import org.springframework.stereotype.Controller; import org.springframework.validation.BindingResult; import org.springframework.validation.annotation.Validated; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.ResponseBody; import com.valid.pojo.People; import com.valid.pojo.Person; @Controller public class FirstController { @RequestMapping("/addPeople") public @ResponseBody String addPeople(@Validated People p,BindingResult result,@Validated Person p2,BindingResult result2) { if(result.hasErrors()) { return "0"; } if(result2.hasErrors()) { return "-1"; } return "1"; } }
補充:主要區(qū)別
在檢驗Controller的入?yún)⑹欠穹弦?guī)范時,使用@Validated或者@Valid在基本驗證功能上沒有太多區(qū)別。但是在分組、注解地方、嵌套驗證等功能上兩個有所不同:
@Validated | @Valid | |
分組 | 提供分組功能,可在入?yún)Ⅱ炞C時,根據(jù)不同的分組采用不同的驗證機制。 | 無分組功能 |
可注解位置 | 可以用在類型、方法和方法參數(shù)上。但是不能用在成員屬性上 | 可以用在方法、構(gòu)造函數(shù)、方法參數(shù)和成員屬性上(兩者是否能用于成員屬性上直接影響能否提供嵌套驗證的功能) |
嵌套驗證 | 用在方法入?yún)⑸蠠o法單獨提供嵌套驗證功能。 不能用在成員屬性上。 也無法提供框架進行嵌套驗證。 能配合嵌套驗證注解@Valid進行嵌套驗證。 | 用在方法入?yún)⑸蠠o法單獨提供嵌套驗證功能。 能夠用在成員屬性上,提示驗證框架進行嵌套驗證。 能配合嵌套驗證注解@Valid進行嵌套驗證。 |
總結(jié)
到此這篇關(guān)于spring注解之@Valid和@Validated的區(qū)分總結(jié)的文章就介紹到這了,更多相關(guān)@Valid和@Validated區(qū)分內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
淺談Java中Collection和Collections的區(qū)別
下面小編就為大家?guī)硪黄獪\談Java中Collection和Collections的區(qū)別。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2016-08-08SpringCloud Stream消息驅(qū)動實例詳解
這篇文章主要介紹了SpringCloud Stream消息驅(qū)動的相關(guān)知識,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2021-03-03macOS中搭建Java8開發(fā)環(huán)境(基于Intel?x86?64-bit)
這篇文章主要介紹了macOS中搭建Java8開發(fā)環(huán)境(基于Intel?x86?64-bit)?的相關(guān)資料,需要的朋友可以參考下2022-12-12java基于servlet使用組件smartUpload實現(xiàn)文件上傳
這篇文章主要介紹了java基于servlet使用組件smartUpload實現(xiàn)文件上傳,具有一定的參考價值,感興趣的小伙伴們可以參考一下2016-10-10Java二維碼登錄流程實現(xiàn)代碼(包含短地址生成,含部分代碼)
近年來,二維碼的使用越來越風生水起,本篇文章主要介紹了Java二維碼登錄流程實現(xiàn)代碼,其中包含短地址生成,有興趣的可以了解一下。2016-12-12SpringBoot項目攔截器獲取Post方法的請求body實現(xiàn)
本文主要介紹了SpringBoot項目攔截器獲取Post方法的請求body,文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2022-01-01java使用定時器實現(xiàn)監(jiān)聽數(shù)據(jù)變化
這篇文章主要為大家詳細介紹了Java如何使用定時器監(jiān)聽數(shù)據(jù)變化,當滿足某個條件時(例如沒有數(shù)據(jù)更新)自動執(zhí)行某項任務(wù),有興趣的可以了解下2023-11-11SpringBoot中MockMVC單元測試的實現(xiàn)
Mock是一種用于模擬和替換類的對象的方法,以便在單元測試中獨立于外部資源進行測試,本文主要介紹了SpringBoot中MockMVC單元測試的實現(xiàn),具有應(yīng)該的參考價值,感興趣的可以了解一下2024-02-02