JSR303 校验扩展(分组、按顺序校验)
1.在spring MVC 项目中使用JSR303 校验数据合法性,一般情况下使用方法为
(1)在接受数据的实体使用注解标添加校验规则
package com.hzsj.wechatdto; import org.hibernate.validator.constraints.Length;
import org.hibernate.validator.constraints.NotBlank; public class MemberApplyDto {
@NotBlank(message="注册号不能为空")
@Length(max=6,min=6,message="注册号必须为6位")
private String registerId; @NotBlank(message="姓名不能为空")
@Length(max=50,message="长度不能超过50个字符")
private String name; @NotBlank(message="选择性别")
private String gender; @NotBlank(message="请填写身份证号码")
@Length(max = 18,message="身份证号码不能超过18个字符")
private String cardNo; @NotBlank(message="请选择省")
private String province; @NotBlank(message="请选择市")
private String cityName; @NotBlank(message="请选择县")
private String countyName; @NotBlank(message="请填写详细地址")
@Length(max=50,message="不能超过50个字符")
private String detailAddress; @NotBlank(message="请选择婚姻状况")
private String marriage; @NotBlank(message="请填写公司名称")
@Length(max=30,message="公司名称不能超过30个字符")
private String companyName; @NotBlank(message="请填写公司电话")
@Length(max=20,message="公司电话不能超过20个字符")
private String companyTel; @NotBlank(message="请填写公司地址")
@Length(max=50,message="公司地址不能超过50个字符")
private String companyAddress; @NotBlank(message="请填写个人简历")
@Length(max=200,message="个人简历不能超过200字符")
private String persomResume; @NotBlank(message="请选择渠道平台")
private String channelType; @NotBlank(message="请填写保荐人")
@Length(max=20,message="保荐人不能超过20个字符")
private String recommend; @NotBlank(message="uuidCode不能为空")
private String uuidCode; public String getRegisterId() {
return registerId;
} public void setRegisterId(String registerId) {
this.registerId = registerId;
} public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} public String getGender() {
return gender;
} public void setGender(String gender) {
this.gender = gender;
} public String getCardNo() {
return cardNo;
} public void setCardNo(String cardNo) {
this.cardNo = cardNo;
} public String getProvince() {
return province;
} public void setProvince(String province) {
this.province = province;
} public String getCityName() {
return cityName;
} public void setCityName(String cityName) {
this.cityName = cityName;
} public String getCountyName() {
return countyName;
} public void setCountyName(String countyName) {
this.countyName = countyName;
} public String getDetailAddress() {
return detailAddress;
} public void setDetailAddress(String detailAddress) {
this.detailAddress = detailAddress;
} public String getMarriage() {
return marriage;
} public void setMarriage(String marriage) {
this.marriage = marriage;
} public String getCompanyName() {
return companyName;
} public void setCompanyName(String companyName) {
this.companyName = companyName;
} public String getCompanyTel() {
return companyTel;
} public void setCompanyTel(String companyTel) {
this.companyTel = companyTel;
} public String getCompanyAddress() {
return companyAddress;
} public void setCompanyAddress(String companyAddress) {
this.companyAddress = companyAddress;
} public String getPersomResume() {
return persomResume;
} public void setPersomResume(String persomResume) {
this.persomResume = persomResume;
} public String getChannelType() {
return channelType;
} public void setChannelType(String channelType) {
this.channelType = channelType;
} public String getRecommend() {
return recommend;
} public void setRecommend(String recommend) {
this.recommend = recommend;
} public String getUuidCode() {
return uuidCode;
} public void setUuidCode(String uuidCode) {
this.uuidCode = uuidCode;
} }
(2)在Controller中使用BindResult 接收校验的结果
@RequestMapping(value="/apply",method=RequestMethod.POST)
@ResponseBody
public ResultVo memberApply(@Valid MemberApplyDto dto,BindingResult bindingResult,Errors errors){
ResultVo<Object> resultVo = new ResultVo<>();
if(errors.hasErrors()){
List<FieldError> errorsList = bindingResult.getFieldErrors();
Map<String, String> map = new HashMap<>();
for(FieldError fieldError:errorsList){
map.put(fieldError.getField(), fieldError.getDefaultMessage());
}
resultVo.setCode(StatusEnums.DATAVALID_ERROR.getCode());
resultVo.setData(map);
resultVo.setMsg(StatusEnums.DATAVALID_ERROR.getMsg());
return resultVo;
}
LoginVo vo = memberApplyService.submitApply(dto);
resultVo.setCode(StatusEnums.SUCCESS.getCode());
resultVo.setMsg(StatusEnums.SUCCESS.getMsg());
resultVo.setData(vo);
return resultVo;
}
2.如果没有特殊需求的情况下使用上面的校验即可。但是遇到其他情况上面的校验就不能满足或者不能灵活应对了。例如
(1)实体中的字段校验按照顺序进行,如果第一个字段校验失败,则接下来的校验不再进行。
(2)实体的字段校验按情况分类,分组校验,再不同方法中校验的字段和顺序不同。
(3)自定义校验规则。
刚好今天遇到上面的三种情况,接下来用实例一一解答。针对上面的需求,JSR303校验中专门提供了group (验证规则所属组)和 @GroupSequence(验证组的顺序) 来实现。
我的需求是表单中的字段按个按顺序校验,如果有前面的字段校验失败,则中断校验并返回校验结果。我的表单中有十几个字段,我的想法是为每个字段定义一个组,然后按照组的顺序进行校验(如果字段很多会比较麻烦,暂时不知道有什么更好的办法, 如果有人知道的话请指教。)于是上面的实体类就变成了下面的样子
package com.hzsj.wechatdto; import java.io.Serializable; import org.hibernate.validator.constraints.Length;
import org.hibernate.validator.constraints.NotBlank; import com.hzsj.common.util.annotation.IsUndefined;
public class MemberApplyDto implements Serializable {
/**
*
*/
private static final long serialVersionUID = -7063091764413674200L; @NotBlank(message="注册号不能为空",groups={Default.class})
@Length(max=6,min=6,message="注册号必须为6位",groups={Default.class})
private String registerId; @NotBlank(message="请填写姓名",groups={Validate1.class})
@Length(max=50,message="长度不能超过50个字符",groups={Validate1.class})
private String name; @NotBlank(message="选择性别",groups={Validate2.class})
private String gender; @NotBlank(message="请填写身份证号码",groups={Validate3.class})
@Length(max = 18,message="身份证号码不能超过18个字符",groups={Validate3.class})
private String cardNo; @NotBlank(message="请选择户籍地区",groups={Validate4.class})
@IsUndefined(message="请选择户籍地区",groups={Validate4.class})
private String province; @NotBlank(message="请选择户籍地区",groups={Validate5.class})
@IsUndefined(message="请选择户籍地区",groups={Validate5.class})
private String cityName; @NotBlank(message="请选择户籍地区",groups={Validate6.class})
@IsUndefined(message="请选择户籍地区",groups={Validate6.class})
private String countyName; @NotBlank(message="请填写详细地址",groups={Validate7.class})
@Length(max=50,message="不能超过50个字符",groups={Validate7.class})
private String detailAddress; @NotBlank(message="请选择婚姻状况",groups={Validate8.class})
private String marriage; @NotBlank(message="请填写公司名称",groups={Validate9.class})
@Length(max=30,message="公司名称不能超过30个字符",groups={Validate9.class})
private String companyName; @NotBlank(message="请填写公司电话",groups={Validate10.class})
@Length(max=20,message="公司电话不能超过20个字符",groups={Validate10.class})
private String companyTel; @NotBlank(message="请填写公司地址",groups={Validate11.class})
@Length(max=50,message="公司地址不能超过50个字符",groups={Validate11.class})
private String companyAddress; @NotBlank(message="请填写个人履历",groups={Validate12.class})
@Length(max=200,message="个人履历不能超过200字符",groups={Validate12.class})
private String persomResume; @NotBlank(message="请选择渠道平台",groups={Validate13.class})
private String channelType; @NotBlank(message="请填写保荐人",groups={Validate14.class})
@Length(max=20,message="保荐人不能超过20个字符",groups={Validate14.class})
private String recommend; @NotBlank(message="uuidCode不能为空",groups={Default.class})
private String uuidCode; public String getRegisterId() {
return registerId;
} public void setRegisterId(String registerId) {
this.registerId = registerId;
} public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} public String getGender() {
return gender;
} public void setGender(String gender) {
this.gender = gender;
} public String getCardNo() {
return cardNo;
} public void setCardNo(String cardNo) {
this.cardNo = cardNo;
} public String getProvince() {
return province;
} public void setProvince(String province) {
this.province = province;
} public String getCityName() {
return cityName;
} public void setCityName(String cityName) {
this.cityName = cityName;
} public String getCountyName() {
return countyName;
} public void setCountyName(String countyName) {
this.countyName = countyName;
} public String getDetailAddress() {
return detailAddress;
} public void setDetailAddress(String detailAddress) {
this.detailAddress = detailAddress;
} public String getMarriage() {
return marriage;
} public void setMarriage(String marriage) {
this.marriage = marriage;
} public String getCompanyName() {
return companyName;
} public void setCompanyName(String companyName) {
this.companyName = companyName;
} public String getCompanyTel() {
return companyTel;
} public void setCompanyTel(String companyTel) {
this.companyTel = companyTel;
} public String getCompanyAddress() {
return companyAddress;
} public void setCompanyAddress(String companyAddress) {
this.companyAddress = companyAddress;
} public String getPersomResume() {
return persomResume;
} public void setPersomResume(String persomResume) {
this.persomResume = persomResume;
} public String getChannelType() {
return channelType;
} public void setChannelType(String channelType) {
this.channelType = channelType;
} public String getRecommend() {
return recommend;
} public void setRecommend(String recommend) {
this.recommend = recommend;
} public String getUuidCode() {
return uuidCode;
} public void setUuidCode(String uuidCode) {
this.uuidCode = uuidCode;
} public interface Validate1{};
public interface Validate2{};
public interface Validate3{};
public interface Validate4{};
public interface Validate5{};
public interface Validate6{};
public interface Validate7{};
public interface Validate8{};
public interface Validate9{};
public interface Validate10{};
public interface Validate11{};
public interface Validate12{};
public interface Validate13{};
public interface Validate14{};
public interface Default{};
}
其中特别说明:实体类中的这些 interface 用来定义一个验证组,类似一个标识。然后为每个字段指定相应的验证组,其余字段使用默认的验证组。
接下来声明一个验证序列,指定这个序列需要验证哪些组和验证的顺序。
package com.hzsj.wechatdto;
import javax.validation.GroupSequence;
@GroupSequence(value={MemberApplyDto.Validate1.class,
MemberApplyDto.Validate2.class,
MemberApplyDto.Validate3.class,
MemberApplyDto.Validate4.class,
MemberApplyDto.Validate5.class,
MemberApplyDto.Validate6.class,
MemberApplyDto.Validate7.class,
MemberApplyDto.Validate8.class,
MemberApplyDto.Validate9.class,
MemberApplyDto.Validate10.class,
MemberApplyDto.Validate11.class,
MemberApplyDto.Validate12.class,
MemberApplyDto.Validate13.class,
MemberApplyDto.Validate14.class,
MemberApplyDto.Default.class,
})
public interface ApplySequence {
}
我指定的是验证所有组,并按照组的顺序验证。如果在某些情况下只需要验证其中部分字段的话,可重新定义一个验证序列,在接下的Controller中去使用这个序列。
@RequestMapping(value="/apply",method=RequestMethod.POST)
@ResponseBody
public ResultVo memberApply(@Validated({ApplySequence.class}) MemberApplyDto dto,BindingResult bindingResult,Errors errors){
ResultVo<Object> resultVo = new ResultVo<>();
if(errors.hasErrors()){
List<FieldError> errorsList = bindingResult.getFieldErrors();
Map<String, String> map = new HashMap<>();
for(FieldError fieldError:errorsList){
map.put(fieldError.getField(), fieldError.getDefaultMessage());
}
resultVo.setCode(StatusEnums.DATAVALID_ERROR.getCode());
resultVo.setData(map);
resultVo.setMsg(StatusEnums.DATAVALID_ERROR.getMsg());
return resultVo;
}
LoginVo vo = memberApplyService.submitApply(dto);
resultVo.setCode(StatusEnums.SUCCESS.getCode());
resultVo.setMsg(StatusEnums.SUCCESS.getMsg());
resultVo.setData(vo);
return resultVo;
}
在Controller中需要的注意的是将原来的@Valid 替换成@Validated 。同时指定了我所需要的验证序列是按照自己定义的验证序列。
@Valid是javax.validation里的。
@Validated是@Valid 的一次封装,是Spring提供的校验机制使用。
相比@Valid @Validated 提供了几个新功能
(1)可以通过groups对验证进行分组
(2)按照序列组来验证
(3)验证多个实体
到此基本实现了按照顺序按个验证字段的合法性,但是同时发现了另外的一种情况,前端字段为空的时候会传过来的undefined,导致原来的验证规则失效。所有我们需要自己去定义一个验证规则去验证undefined。上面的实体使用的 @IsUndefined 就是我自行定义的。
首先定义一个注解,同时指定实现校验规则的类 validatedBy = {UndefinedValiadator.class}
package com.hzsj.common.util.annotation; import java.lang.annotation.Documented;
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.TYPE, ElementType.METHOD, ElementType.PARAMETER,ElementType.CONSTRUCTOR,ElementType.ANNOTATION_TYPE,ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Constraint(validatedBy = {UndefinedValiadator.class})
public @interface IsUndefined { //提示信息
String message() default ""; Class<?>[] groups() default {}; Class<? extends Payload>[] payload() default {};
}
其次,实现这个校验规则
package com.hzsj.common.util.annotation; import javax.validation.ConstraintValidator;
import javax.validation.ConstraintValidatorContext; import org.springframework.util.StringUtils; public class UndefinedValiadator implements ConstraintValidator<IsUndefined,String>{ @Override
public void initialize(IsUndefined constraintAnnotation) {
} @Override
public boolean isValid(String value, ConstraintValidatorContext context) {
if(StringUtils.isEmpty(value)){
return false;
}
if("undefined".equals(value)){
return false;
}else{
return true;
}
} }
至此完成了一个自定义的规则,可以在自己的实体类中去使用了
JSR303 校验扩展(分组、按顺序校验)的更多相关文章
- ESLint – 可扩展的 JavaScript & JSX 校验工具
ESLint 是一个开源的 JavaScript 代码校验工具,最初是由 Nicholas C. Zakas 在2013年创建的.经常被用来发现问题的模式或代码,不符合特定的风格准则. ESLint ...
- Solon详解(六)- Solon的校验扩展框架使用与扩展
Solon详解系列文章: Solon详解(一)- 快速入门 Solon详解(二)- Solon的核心 Solon详解(三)- Solon的web开发 Solon详解(四)- Solon的事务传播机制 ...
- 点分十进制IP校验、转换,掩码校验
/***************************************************************************** * 点分十进制IP校验.转换,掩码校验 * ...
- Struts2 对Action中所有方法进行输入校验、单个方法进行校验
index.jsp: <body> <s:fielderror /> <form action="${pageContext.request.contextPa ...
- 更加灵活的参数校验,Spring-boot自定义参数校验注解
上文我们讨论了如何使用@Min.@Max等注解进行参数校验,主要是针对基本数据类型和级联对象进行参数校验的演示,但是在实际中我们往往需要更为复杂的校验规则,比如注册用户的密码和确认密码进行校验,这个时 ...
- Form的is_valid校验规则及验证顺序
一.验证顺序 查看form下的源码了解顺序 BaseForm为基类,中间包含了is_valid校验方法 @html_safe class BaseForm: ......... self.is_b ...
- EasyUI 扩展自己定义EasyUI校验规则 验证规则(经常使用的)
比如 校验输入框仅仅能录入0-1000之间 最多有2位小数的数字 表单<input type="text" id="rate" name="ra ...
- 扩展对EasyUI的校验规则
var myReg = RegExp(/[(\*)(\|)(\\)(\:)(\")(\/)(\<)(\>)(\?)]+/); $.extend($.fn.validatebox. ...
- easyui validate -- radio、checkbox 校验扩展,事件域名
事件域名: $(dom).on('click.myNameSpace',function(){ ... }),其中‘.myNameSpace’便是域名: 目前作用:$(dom).off('click. ...
随机推荐
- linux运维、架构之路-K8s滚动更新及回滚
一.滚动更新 应用程序一次只更新一小部分副本,更新成功后,再更新更多的副本,最终完成所有副本的更新. 滚动更新的优点:零停机,整个更新过程始终有副本在运行,从而保证了业务的连续性. 1. ...
- TTTTTTTTTTTTTTTTTTTTT POJ 3690 0与* 二维哈希 模板 +multiset
Constellations Time Limit: 3000MS Memory Limit: 65536K Total Submissions: 5923 Accepted: 1164 De ...
- Spring配置文件beans标签报错问题解决
因为有很多配置是复制过来的,附带的很多注释的格式会导致报错,所以可以要试试把注释去掉,只有配置文件的话可能就不会报错了.
- 「HAOI2016」食物链
题目链接 解题思路 简单的DAG上DP即可. 参考程序 #include <bits/stdc++.h> using namespace std; const int Maxn = 100 ...
- AtCoder AGC022C Remainder Game (图论)
题目链接 https://atcoder.jp/contests/agc022/tasks/agc022_c 题解 大水题一道 就他给的这个代价,猜都能猜到每个数只能用一次 仔细想想,我们肯定是按顺序 ...
- Unity3D_(插件)DOTween动画插件
使用DOTween动画插件来实现物体的移动动画 Learn 一.DOTween插件对变量的动画 二.控制Cube和UI面板的动画 三.动画的快捷播放方式 四.动画的前放和后放 五.From Tween ...
- ES6迭代器和生成器
一.迭代器 JavaScript 原有的表示"集合"的数据结构,主要是数组(Array)和对象(Object),ES6 又添加了Map和Set.这样就需要一种统一的接口机制,来处理 ...
- ajax 调用 webService
前台脚本<script type="text/javascript"> $(function () { $.ajax({ type: "POST", ...
- jenkins 内置变量
Jenkins 有一些内置的变量可以使用.主要是: 邮件的配置变量,可以在发送邮件的时候使用. 环境变量 1. 邮件的配置变量 ${GIT_BRANCH} - build 的 Git 分支 ${FIL ...
- C语言转义字符表和ASCII码表
主要参考 http://www.51hei.com/mcu/4342.html 以及 https://www.cnblogs.com/jason207489550/p/6663444.html