Проверка двойных и плавающих значений с помощью Hibernate Validator - bean валидация

Я ищу способ проверки поля java.lang.Double в команде Spring bean для его максимального и минимального значений (значение должно лежать между заданным диапазоном значений), например

public final class WeightBean
{
     @Max(groups={ValidationGroup.class}, value=Double.MAX_VALUE, message="some key or default message")
     @Min(groups={ValidationGroup.class}, value=1D, message="some key or default message")
     private Double txtWeight;  //Getter and setter.

     public interface ValidationGroup{}         
}

Но оба @Max и @Min не может принимать значение java.lang.Double.

Обратите внимание, что double и float не поддерживаются из-за ошибок округления (некоторые провайдеры могут обеспечить некоторую аппроксимативную поддержку)

Итак, каков способ проверки таких полей?

Я работаю с Spring 3.2.0 и Hibernate Validator 4.3.1 CR1.

Ответ 1

Вы можете использовать аннотацию, но вы можете получить ложные результаты. Это общая проблема с удвоениями и imo во многих случаях. Следует избегать Double_s. Может быть, переключение на другой тип - лучшее решение? Например, BigDecimal?

Ответ 2

Если вы переключились на BigDecimal (или BigInteger), вы можете использовать @DecimalMin или @DecimalMax. Но это все еще не решение для float или double.

Ответ 3

Я избегал типов double и float и реализовал специальный валидатор, который мог бы проверить значение BigDecimal на основе точности и масштаба.

Дескриптор ограничения.

package constraintdescriptor;

import constraintvalidator.BigDecimalRangeValidator;
import java.lang.annotation.Documented;
import static java.lang.annotation.ElementType.ANNOTATION_TYPE;
import static java.lang.annotation.ElementType.FIELD;
import static java.lang.annotation.ElementType.METHOD;
import java.lang.annotation.Retention;
import static java.lang.annotation.RetentionPolicy.RUNTIME;
import java.lang.annotation.Target;
import javax.validation.Constraint;
import javax.validation.Payload;

@Target({METHOD, FIELD, ANNOTATION_TYPE})
@Retention(RUNTIME)
@Constraint(validatedBy = BigDecimalRangeValidator.class)
@Documented
public @interface BigDecimalRange {
    public String message() default "{java.math.BigDecimal.range.error}";
    public Class<?>[] groups() default {};
    public Class<? extends Payload>[] payload() default {};

    long minPrecision() default Long.MIN_VALUE;
    long maxPrecision() default Long.MAX_VALUE;
    int scale() default 0;
}

Валидатор ограничений.

package constraintvalidator;

import constraintdescriptor.BigDecimalRange;
import java.math.BigDecimal;
import javax.validation.ConstraintValidator;
import javax.validation.ConstraintValidatorContext;

public final class BigDecimalRangeValidator implements ConstraintValidator<BigDecimalRange, Object> {

    private long maxPrecision;
    private long minPrecision;
    private int scale;

    @Override
    public void initialize(final BigDecimalRange bigDecimalRange) {
        maxPrecision = bigDecimalRange.maxPrecision();
        minPrecision = bigDecimalRange.minPrecision();
        scale = bigDecimalRange.scale();
    }

    @Override
    public boolean isValid(final Object object, final ConstraintValidatorContext cvc) {
        boolean isValid = false;

        if (object == null) { // This should be validated by the not null validator (@NotNull).
            isValid = true;
        } else if (object instanceof BigDecimal) {
            BigDecimal bigDecimal = new BigDecimal(object.toString());
            int actualPrecision = bigDecimal.precision();
            int actualScale = bigDecimal.scale();
            isValid = actualPrecision >= minPrecision && actualPrecision <= maxPrecision && actualScale <= scale;

            if (!isValid) {
                cvc.disableDefaultConstraintViolation();
                cvc.buildConstraintViolationWithTemplate("Precision expected (minimun : " + minPrecision + ", maximum : " + maxPrecision + "). Maximum scale expected : " + scale + ". Found precision : " + actualPrecision + ", scale : " + actualScale).addConstraintViolation();
            }
        }

        return isValid;
    }
}

Это может быть расширено и для других типов, когда это необходимо.


И, наконец, в bean свойство типа BigDecimal может быть аннотировано аннотацией @BigDecimalRange следующим образом.

package validatorbeans;

public final class WeightBean {

    @BigDecimalRange(minPrecision = 1, maxPrecision = 33, scale = 2, groups = {ValidationGroup.class}, message = "The precision and the scale should be less than or equal to 35 and 2 respectively.")
    private BigDecimal txtWeight; // Getter and setter.

    public interface ValidationGroup {}
}

Ответ 4

Иногда это удобно в паре с @AssertTrue/@AssertFalse из javax.validation.constraints

public final class WeightBean {
    @NotNull
    private Double txtWeight;  //Getter and setter.

    @AssertTrue
    public boolean getTxtWeightCheck() {
        return txtWeight > 0.1 && txtWeight < 0.9;
    }
}

Ответ 5

Вы также можете использовать @Digits в API-интерфейсе проверки гибернации

@Digits(integer = 10 /*precision*/, fraction = 2 /*scale*/)

Ответ 6

Более новые версии Hibernate Validator (по крайней мере 6.0.17) поддерживают аннотацию @DecimalMin/Max для двойного

Смотрите класс для проверки