简介

JSR-303 是JAVA EE 6 中的一项子规范,叫做Bean Validation,Hibernate Validator 是 Bean Validation 的参考实现。 Hibernate Validator 提供了 JSR 303 规范中所有内置 constraint 的实现,除此之外还有一些附加的 constraint。
下载 JSR 303 – Bean Validation 规范 http://jcp.org/en/jsr/detail?id=303
如果想了解更多有关 Hibernate Validator 的信息,请查看 http://www.hibernate.org/subprojects/validator.html

Bean Validation 中的 constraint

Constraint 详细信息
@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(value) 被注释的元素必须符合指定的正则表达式

Hibernate Validator 附加的 constraint

Constraint 详细信息
@Email 被注释的元素必须是电子邮箱地址
@Length 被注释的字符串的大小必须在指定的范围内
@NotEmpty 被注释的字符串的必须非空
@Range 被注释的元素必须在合适的范围内
  1. 空检查
  2. @Null 验证对象是否为null
  3. @NotNull 验证对象是否不为null, 无法查检长度为0的字符串
  4. @NotBlank 检查约束字符串是不是Null还有被Trim的长度是否大于0,只对字符串,且会去掉前后空格.
  5. @NotEmpty 检查约束元素是否为NULL或者是EMPTY.
  6. Booelan检查
  7. @AssertTrue 验证 Boolean 对象是否为 true
  8. @AssertFalse 验证 Boolean 对象是否为 false
  9. 长度检查
  10. @Size(min=, max=) 验证对象(Array,Collection,Map,String)长度是否在给定的范围之内
  11. @Length(min=, max=) Validates that the annotated string is between min and max included.
  12. 日期检查
  13. @Past 验证 Date Calendar 对象是否在当前时间之前
  14. @Future 验证 Date Calendar 对象是否在当前时间之后
  15. @Pattern 验证 String 对象是否符合正则表达式的规则
  16. 数值检查,建议使用在Stirng,Integer类型,不建议使用在int类型上,因为表单值为“”时无法转换为int,但可以转换为Stirng"",Integernull
  17. @Min 验证 Number String 对象是否大等于指定的值
  18. @Max 验证 Number String 对象是否小等于指定的值
  19. @DecimalMax 被标注的值必须不大于约束中指定的最大值. 这个约束的参数是一个通过BigDecimal定义的最大值的字符串表示.小数存在精度
  20. @DecimalMin 被标注的值必须不小于约束中指定的最小值. 这个约束的参数是一个通过BigDecimal定义的最小值的字符串表示.小数存在精度
  21. @Digits 验证 Number String 的构成是否合法
  22. @Digits(integer=,fraction=) 验证字符串是否是符合指定格式的数字,interger指定整数精度,fraction指定小数精度。
  23. @Range(min=, max=) 检查数字是否介于minmax之间.
  24. @Range(min=10000,max=50000,message="range.bean.wage")private BigDecimal wage;
  25. @Valid 递归的对关联对象进行校验, 如果关联对象是个集合或者数组,那么对其中的元素进行递归校验,如果是一个map,则对其中的值部分进行校验.(是否进行递归验证)
  26. @CreditCardNumber信用卡验证
  27. @Email 验证是否是邮件地址,如果为null,不进行验证,算通过验证。
  28. @ScriptAssert(lang= ,script=, alias=)
  29. @URL(protocol=,host=, port=,regexp=, flags=)

使用步骤

导包

  1. <dependency>
  2. <groupId>org.hibernate.validator</groupId>
  3. <artifactId>hibernate-validator</artifactId>
  4. <version>6.0.16.Final</version>
  5. <scope>compile</scope>
  6. </dependency>

配置

  1. 在controller的类上加上@Validated注解
  1. @Validated
  2. @RestController
  3. public class ClauseInfoController {}
  1. 当时基本类型请求参数
  1. @GetMapping("/xxxx")
  2. public ObjectRestResponse<String>
  3. queryAllStaffList(@RequestParam @NotBlank(message = "部门ID不能为空") String departmentKey) {
  4. List<ResultDataVO> staff = clauseInfoService.queryAllStaffList(departmentKey);
  5. return ObjectRestResponse.success().data(staff);
  6. }
  1. 对象参数
  1. @PostMapping("/xxxxx")
  2. public ObjectRestResponse<String>
  3. addSppiTableList(@RequestBody @Valid ProjectClauseParam projectClauseParam) {
  4. clauseInfoService.addSppiTableList(projectClauseParam);
  5. return ObjectRestResponse.success().data(ConfigConstant.RESULT_SUCCESS);
  6. }
  7. @Data
  8. public class ProjectClauseParam {
  9. @NotBlank(message = "项目名称不能为空")
  10. @Length(max = 100, message = "项目名称长度不能超过100")
  11. private String projectName;
  12. @NotBlank(message = "参与人不能为空")
  13. private String participantKey;
  14. }

异常处理

  1. import com.clamc.common.msg.ObjectRestResponse;
  2. import org.apache.commons.lang3.StringUtils;
  3. import org.springframework.context.support.DefaultMessageSourceResolvable;
  4. import org.springframework.http.HttpStatus;
  5. import org.springframework.validation.BindException;
  6. import org.springframework.web.bind.annotation.ControllerAdvice;
  7. import org.springframework.web.bind.annotation.ExceptionHandler;
  8. import org.springframework.web.bind.annotation.ResponseBody;
  9. import javax.servlet.http.HttpServletRequest;
  10. import javax.validation.ConstraintViolationException;
  11. import java.util.List;
  12. import java.util.stream.Collectors;
  13. /**
  14. * @author guojianfeng.
  15. * @date 2019/11/7
  16. */
  17. @ControllerAdvice
  18. @ResponseBody
  19. public class GlobalExceptionHandler {
  20. @ExceptionHandler({ConstraintViolationException.class,BindException.class})
  21. public ObjectRestResponse<String> exceptionHandler1(HttpServletRequest request, Exception exception) {
  22. ObjectRestResponse response = new ObjectRestResponse();
  23. response.setStatus(HttpStatus.INTERNAL_SERVER_ERROR.value());
  24. if (exception instanceof ConstraintViolationException){
  25. ConstraintViolationException ex = (ConstraintViolationException) exception;
  26. String errorMsg = ex.getConstraintViolations().iterator().next().getMessage();
  27. response.setMessage(errorMsg);
  28. }else {
  29. /*注意:此处的BindException 是 Spring 框架抛出的Validation异常*/
  30. BindException ex = (BindException) exception;
  31. /*抛出异常可能不止一个 输出为一个List集合*/
  32. List<String> errors = ex.getAllErrors().stream().map(DefaultMessageSourceResolvable::getDefaultMessage).collect(Collectors.toList());
  33. /*获取异常信息*/
  34. String errorMsg = StringUtils.join(errors);
  35. response.setMessage(errorMsg);
  36. }
  37. return response.rel(false);
  38. }
  39. }

自动定义参数校验

定义注解

  1. @Target({METHOD, FIELD, ANNOTATION_TYPE, CONSTRUCTOR, PARAMETER})
  2. @Retention(RUNTIME)
  3. @Documented
  4. @Constraint(validatedBy = {IsMobileValidator.class})
  5. public @interface IsMobile {
  6. boolean required() default true;
  7. String message() default "手机号码格式错误";
  8. Class<?>[] groups() default {};
  9. Class<? extends Payload>[] payload() default {};
  10. }

继承 ConstraintValidator

ConstraintValidator

  • IsMobile 是我们定义的注解
  • String 是这个注解修改时类型
  1. import javax.validation.ConstraintValidator;
  2. import javax.validation.ConstraintValidatorContext;
  3. public class IsMobileValidator implements ConstraintValidator<IsMobile, String> {
  4. private boolean required = false;
  5. @Override
  6. public void initialize(IsMobile constraintAnnotation) {
  7. required = constraintAnnotation.required();
  8. }
  9. @Override
  10. public boolean isValid(String value, ConstraintValidatorContext context) {
  11. if (required) {
  12. return ValidatorUtil.isMobile(value);
  13. } else {
  14. if (StringUtils.isEmpty(value)) {
  15. return true;
  16. } else {
  17. return ValidatorUtil.isMobile(value);
  18. }
  19. }
  20. }
  21. }
  22. -----------------------------------------------------------------------------------
  23. public class ValidatorUtil {
  24. private static final Pattern mobile_pattern = Pattern.compile("1\\d{10}");
  25. public static boolean isMobile(String src) {
  26. if (StringUtils.isEmpty(src)) {
  27. return false;
  28. }
  29. Matcher m = mobile_pattern.matcher(src);
  30. return m.matches();
  31. }
  32. }