今天看啥  ›  专栏  ›  小胖学编程

校验框架——validation的使用

小胖学编程  · 简书  ·  · 2019-04-29 16:45

java bean validation 参数验证

Java Bean Validation(参数校验) 最佳实践

后台如何对参数进行校验?比如参数不能为null,email那么必须符合email的格式,如果手动进行if判断或者写正则表达式判断无疑发效率太慢。那么有没有现成的框架让我们使用呢?

1. validation bean介绍

1.1 什么叫做validation bean

validation bean是基于JSR-303标准开发出来的,使用注解方式实现,及其方便。是这只是一个接口,没有具体实现.Hibernate Validator是一个hibernate独立的包,可以直接引用,他实现了validation bean同时有做了扩展,比较强大 ,实现图如下:

hibernate validator

1.2 注解

validation bean的注解

@NotEmpty、@NotNull、@NotBlank的区别

  • @NotEmpty 用在集合String上面
  • @NotBlank用在String上面(不能用在基本数据类型上面)
  • @NotNull用在基本数据类型上面

hibernate validator中Digits如何检查大小数点是否只有2位数字。

class Item {
   @Digits(integer = 6, fraction = 2, message = "{javax.validation.constraints.Digits.message}")
   private BigDecimal amount;
}

其中integer和fraction是强制的,并且消息是可选参数。

正则表达式中,如何允许【空格】或者【空串】

public class User {
    
    @Pattern(regexp = "(^0|1|[ ]+$)|(^$)|",message = "信息不符")
    private String name;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
}

[ ]+表示允许一个或者多个空格,而^$表示是空串。

1.3 验证方法

maven依赖

<dependency>
            <groupId>javax.validation</groupId>
            <artifactId>validation-api</artifactId>
            <version>1.1.0.Final</version>
        </dependency>
        <!-- hibernate validator-->
        <dependency>
            <groupId>org.hibernate</groupId>
            <artifactId>hibernate-validator</artifactId>
            <version>5.2.0.Final</version>
        </dependency>

核心方法-验证

public class ValidateUtil {

    private static Validator validator; // thread-safe

    static {
        ValidatorFactory factory = Validation.buildDefaultValidatorFactory();
        validator = factory.getValidator();
    }

    public static <T> void validate(T t) throws ValidateException {
        if (null == t) {
            throw new ValidateException("300", "验证数据不能为空");
        }
        List<String> errorList = new ArrayList<String>();
        Set<ConstraintViolation<T>> constraintViolations = validator.validate(t);
        for (ConstraintViolation<T> constraintViolation : constraintViolations) {
            errorList.add(constraintViolation.getMessage());
        }
        if ((null != errorList) && !errorList.isEmpty()) {
            throw new ValidateException("301", StringUtils.join(errorList, ","));
        }
    }
}

测试方法

   public static void main(String[] args) {
        User user = new User();
        try {
            ValidateUtil.validate(user);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
StringUtils.join方法

2. 自定义验证注解

1. 参数枚举

/**
 * 枚举类
 */
public enum  CaseMode {
    UPPER,LOWER
}

2. 声明注解

此处使用@Constraint注解处理对象的逻辑。

import static java.lang.annotation.ElementType.*;
import static java.lang.annotation.RetentionPolicy.*;

import java.lang.annotation.Documented;
import java.lang.annotation.Retention;
import java.lang.annotation.Target;

import javax.validation.Constraint;
import javax.validation.Payload;

@Target({METHOD, FIELD, ANNOTATION_TYPE})
@Retention(RUNTIME)
@Constraint(validatedBy = CheckCaseValidator.class)
@Documented
public @interface CheckCase {

    String message() default "{com.mycompany.constraints.checkcase}";

    Class<?>[] groups() default {};

    Class<? extends Payload>[] payload() default {};

    CaseMode value();

}

3. 注解逻辑

import com.creditease.eaclient.bg.util.StringUtils;

import javax.validation.ConstraintValidator;
import javax.validation.ConstraintValidatorContext;

public class CheckCaseValidator implements ConstraintValidator<CheckCase, String> {

    private CaseMode caseMode;

    //获取传入的CaseMode对象
    @Override
    public void initialize(CheckCase constraintAnnotation) {
        this.caseMode = constraintAnnotation.value();
    }

    @Override
    public boolean isValid(String value, ConstraintValidatorContext context) {
        if (StringUtils.isBlank(value)) {
            return true;
        }
        if (caseMode == CaseMode.UPPER) {
            return value.equals(value.toUpperCase());
        } else {
            return value.equals(value.toLowerCase());
        }
    }
}

4. 使用方法

public class User {

    @CheckCase(value = CaseMode.UPPER, message = "name需要大写")
    private String name;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
}

5. 运行结果

运行结果

@Constraint注解配合自定义验证类型注解的开发

自定义注解逻辑处理类由于实现了ConstraintValidator接口,所以它默认被spring管理成bean,所以可以在这个逻辑处理类里面用@Autowiredu或者@Resources注入别的服务,而且不用在类上面用@Compent注解成spring的bean。

自定义验证类型注解类里面由于是用于验证数据,一般在里面加上

String message() default "用户不存在或者不属于当前组织";

Class<?>[] groups() default {};

Class<? extends Payload>[] payload() default {};



原文地址:访问原文地址
快照地址: 访问文章快照