概述:在java开发时,由于分层的原因(表现层-控制层-业务层-数据持久层),有时候需要对传入的Javabean进行校验,如果过多的校验会导致比较繁琐,做重复的工作,下面将介绍Bean Validation技术,

该技术是利用注解的方式,在javabean代码内部,利用注解实现校验,这样会将繁琐的工作变得简单。

注:在阅读如下知识之前,要对注解有一些了解。

来一个简单的Bean Validation实现 ,热热身:

  1. package validation;
  2. import java.util.Set;
  3. import javax.validation.ConstraintViolation;
  4. import javax.validation.Validation;
  5. import javax.validation.Validator;
  6. import javax.validation.ValidatorFactory;
  7. import javax.validation.constraints.NotNull;
  8. import javax.xml.bind.ValidationException;
  9. //import validation.vo.Person;
  10. /**
  11. * @ClassName: Test1
  12. * @Description: TODO
  13. * @author zhangyy
  14. * @date 2015-7-30 上午11:44:15
  15. */
  16. public class Test1 {
  17. public static void main(String [] args ){
  18. Person person = new Person();
  19. try {
  20. Test1.validate(person);
  21. } catch (ValidationException e) {
  22. System.out.println(e.getMessage());  //输出结果是:用户名不能为空
  23. }
  24. }
  25. public static <T>  void validate(T t) throws ValidationException {
  26. ValidatorFactory vFactory = Validation.buildDefaultValidatorFactory();
  27. Validator validator = vFactory.getValidator();
  28. Set<ConstraintViolation<T>> set =  validator.validate(t);
  29. if(set.size()>0){
  30. StringBuilder validateError = new StringBuilder();
  31. for(ConstraintViolation<T> val : set){
  32. validateError.append(val.getMessage());
  33. }
  34. throw new ValidationException(validateError.toString());
  35. }
  36. }
  37. }
  38. class Person{
  39. @NotNull(message="用户名不能为空")  //此处为校验注解
  40. private String username;
  41. public String getUsername() {
  42. return username;
  43. }
  44. public void setUsername(String username) {
  45. this.username = username;
  46. }
  47. }

上面的代码用到了@NotNull()注解进行校验,一个完整的校验步骤包括如下四个步骤:

1.约束注解的定义

2.约束验证规则(约束验证器)

3.约束注解的声明

4.约束验证流程

下面用代码来进行详细的解释:

1.约束的定义:

Bean Vlidation技术提供了一些内置的约束定义,还可以自定义;下面是内置的:

约束注解名称约束注解说明

@Null验证对象是否为空
@NotNull验证对象是否为非空
@AssertTrue验证 Boolean 对象是否为 true
@AssertFalse验证 Boolean 对象是否为 false
@Min验证 Number 和 String 对象是否大等于指定的值
@Max验证 Number 和 String 对象是否小等于指定的值
@DecimalMin验证 Number 和 String 对象是否大等于指定的值,小数存在精度
@DecimalMax验证 Number 和 String 对象是否小等于指定的值,小数存在精度
@Size验证对象(Array,Collection,Map,String)长度是否在给定的范围之内
@Digits验证 Number 和 String 的构成是否合法
@Past验证 Date 和 Calendar 对象是否在当前时间之前
@Future验证 Date 和 Calendar 对象是否在当前时间之后
@Pattern验证 String 对象是否符合正则表达式的规则

自定义约束的结构如下(其实就是注解的定义):

  1. @Target({ })   // 约束注解应用的目标元素类型(METHOD, FIELD, TYPE, ANNOTATION_TYPE, CONSTRUCTOR, PARAMETER等)
  2. @Retention()   // 约束注解应用的时机
  3. @Constraint(validatedBy ={})  // 与约束注解关联的验证器
  4. public @interface ConstraintName{
  5. String message() default " ";   // 约束注解验证时的输出消息
  6. Class<?>[] groups() default { };  // 约束注解在验证时所属的组别
  7. Class<? extends Payload>[] payload() default { }; // 约束注解的有效负载
  8. }

可以遵循这个格式,去写自己定义的注解(下面有自定义的例子)

2.约束证规则(约束验证器) 

上面的内容是定义约束,当约束定义好了之后,就需要定义一个约束校验规则,其实就是一个校验器。每一个约束定义要对应一个约束校验器。定义约束校验器必须要实现如下的接口:

  1. //约束验证器需要实现该接口
  2. public interface ConstraintValidator<A extends Annotation, T> {
  3. void initialize(A constraintAnnotation);    //初始化验证器
  4. boolean isValid(T value, ConstraintValidatorContext context);   //约束验证的方法 ,这里面实现校验的具体规则
  5. }

下面会有具体的实现约束校验器

3.约束的声明

声明,其实就是讲自定义的注解或者是内置的注解声明在需要校验的字段/方法等上面,该步骤比较简单,如:

  1. @NumberVlidator(message= "体重必须为数字")
  2. private String weight;

 4.验证流程

在实际使用中调用 Validator.validate(JavaBeanInstance) 方法后,Bean Validation 会查找在 JavaBeanInstance上所有的约束声明,对每一个约束调用对应的约束验证器进行验证,最后的结果由约束验证器的 isValid 方法产生,如果该方法返回 true,则约束验证成功,否则验证失败。验证失败的约束将产生约束违规对象(ConstraintViolation 的实例)并放到约束违规列表中。验证完成后所有的验证失败信息均能在该列表中查找并输出

  1. /**
  2. * @throws ValidationException
  3. * @throws ValidationException
  4. * @Description: 校验方法
  5. * @param t 将要校验的对象
  6. * @throws ValidationException
  7. * void
  8. * @throws
  9. */
  10. public static <T> void validate(T t) throws ValidationException{
  11. ValidatorFactory vf = Validation.buildDefaultValidatorFactory();
  12. Validator  validator = vf.getValidator();
  13. Set<ConstraintViolation<T>> set =  validator.validate(t);
  14. if(set.size()>0){
  15. StringBuilder validateError = new StringBuilder();
  16. for(ConstraintViolation<T> val : set){
  17. validateError.append(val.getMessage() + " ;");
  18. }
  19. throw new ValidationException(validateError.toString());
  20. }
  21. }

至此,一个完整的Bean Validation校验已经完成,是不是很简单,如果还是看不懂的话,下面直接上代码,相信看了代码都会明白的,嘿嘿

 

完整的Demo

1.一个javabean对象

  1. package validation.vo;
  2. import java.util.Date;
  3. import javax.validation.constraints.NotNull;
  4. import javax.validation.constraints.Size;
  5. import validation.validate.NumberVlidator;
  6. /**
  7. * @ClassName: Person
  8. * @Description: TODO
  9. * @author zhangyy
  10. * @date 2015-7-30 上午11:46:37
  11. */
  12. public class Person {
  13. @NotNull(message = "用户ID不能为空")
  14. private Integer id;     //应为包装类型,否则不能检测到
  15. @NotNull(message = "test不能为空")
  16. private String test;
  17. @NumberVlidator(message= "体重必须为数字")  //该注解为自定义注解
  18. private String weight;
  19. @NotNull(message = "用户姓名不能为空dd")
  20. @Size(min=1, max=10, message="用户姓名必须是1-10位之间")
  21. private String username;
  22. //省略setter和getter方法
  23. }

2.约束的定义(上面有一个自定义的@NumberVlidator 注解)

  1. package validation.validate;
  2. import java.lang.annotation.Documented;
  3. import java.lang.annotation.ElementType;
  4. import java.lang.annotation.Retention;
  5. import java.lang.annotation.RetentionPolicy;
  6. import java.lang.annotation.Target;
  7. import javax.validation.Constraint;
  8. import javax.validation.Payload;
  9. /**
  10. * @ClassName: NumberVlidator
  11. * @Description: 约束定义
  12. * @author zhangyy
  13. * @date 2015-7-31 上午10:11:14
  14. */
  15. @Target({ElementType.ANNOTATION_TYPE,ElementType.METHOD,ElementType.FIELD})
  16. @Retention(RetentionPolicy.RUNTIME)
  17. @Documented
  18. @Constraint(validatedBy = {NumberVlidatorImpl.class})
  19. public @interface NumberVlidator {
  20. boolean isNumber () default false;
  21. String message() default "该值应该为数字";   // 约束注解验证时的输出消息
  22. Class<?>[] groups() default { };  // 约束注解在验证时所属的组别
  23. Class<? extends Payload>[] payload() default { }; // 约束注解的有效负载
  24. }

3.对上一步的自定义约束进行实现校验规则(校验器)

  1. package validation.validate;
  2. import java.math.BigDecimal;
  3. import javax.validation.ConstraintValidator;
  4. import javax.validation.ConstraintValidatorContext;
  5. /**
  6. * @ClassName: NumberVlidatorImpl
  7. * @Description: 约束验证器
  8. * @author zhangyy
  9. * @date 2015-7-31 上午10:14:44
  10. */
  11. public class NumberVlidatorImpl implements ConstraintValidator<NumberVlidator, String> {
  12. private boolean isNumber;
  13. /**
  14. * <p>Title: 对验证器进行实例化</p>
  15. * @param constraintAnnotation
  16. */
  17. @Override
  18. public void initialize(NumberVlidator constraintAnnotation) {  //初始化
  19. isNumber = constraintAnnotation.isNumber();
  20. }
  21. /**
  22. * <p>Description: 校验的方法</p>
  23. * @param value  需要验证的实例
  24. * @param context 约束执行的上下文环境
  25. * @return
  26. */
  27. @Override
  28. public boolean isValid(String value, ConstraintValidatorContext context) {
  29. if(value==null || value.length()<=0){
  30. return true;
  31. }else{
  32. try {
  33. if(isNumber){
  34. Long.parseLong(value);
  35. }else{
  36. new BigDecimal(value);
  37. }
  38. return true;
  39. } catch (NumberFormatException e) {
  40. return false;
  41. }
  42. }
  43. }
  44. }

4.具体的校验实现(校验工具)

  1. package validation.util;
  2. import java.util.Set;
  3. import javax.validation.ConstraintViolation;
  4. import javax.validation.Validation;
  5. //import javax.validation.ValidationException;
  6. import javax.validation.Validator;
  7. import javax.validation.ValidatorFactory;
  8. import javax.xml.bind.ValidationException;
  9. /**
  10. * @ClassName: VlidationUtil
  11. * @Description: 校验工具类
  12. * @author zhangyy
  13. * @date 2015-7-31 上午10:28:48
  14. */
  15. public class VlidationUtil {
  16. private static Validator validator;
  17. static {
  18. ValidatorFactory vf = Validation.buildDefaultValidatorFactory();
  19. validator = vf.getValidator();
  20. }
  21. /**
  22. * @throws ValidationException
  23. * @throws ValidationException
  24. * @Description: 校验方法
  25. * @param t 将要校验的对象
  26. * @throws ValidationException
  27. * void
  28. * @throws
  29. */
  30. public static <T> void validate(T t) throws ValidationException{
  31. Set<ConstraintViolation<T>> set =  validator.validate(t);
  32. if(set.size()>0){
  33. StringBuilder validateError = new StringBuilder();
  34. for(ConstraintViolation<T> val : set){
  35. validateError.append(val.getMessage() + " ;");
  36. }
  37. throw new ValidationException(validateError.toString());
  38. }
  39. }
  40. }

5.注意: 以上的需要依赖其他的类库,下面是maven的依赖(该步骤不能忘记哦)

  1. <dependency>
  2. <groupId>javax.validation</groupId>
  3. <artifactId>validation-api</artifactId>
  4. <version>1.1.0.Final</version>
  5. </dependency>
  6. <dependency>
  7. <groupId>org.ow2.util.bundles</groupId>
  8. <artifactId>hibernate-validator-4.3.1.Final</artifactId>
  9. <version>1.0.0</version>
  10. </dependency>

6.测试类

  1. package validation;
  2. import javax.xml.bind.ValidationException;
  3. import validation.util.VlidationUtil;
  4. import validation.vo.Person;
  5. /**
  6. * @ClassName: 测试类
  7. * @Description: TODO
  8. * @author zhangyy
  9. * @date 2015-7-30 上午11:44:15
  10. */
  11. public class Test1 {
  12. public static void main(String [] args ){
  13. Person person = new Person();
  14. try {
  15. VlidationUtil.validate(person);
  16. } catch (ValidationException e) {
  17. System.out.println(e.getMessage());
  18. }
  19. //输出结果为:test不能为空 ;用户ID不能为空 ;用户姓名不能为空dd ;
  20. }
  21. }

OK,整个流程结束!

总结:Bean Validation技术除了可以校验一般的数据类型,还支持校验复杂的对象类型,组合类型、等,具体的自行查阅相关资料吧,一般的需求都可以满足的!

04-14 23:10