Ответ 1
Вы можете использовать аннотацию, но вы можете получить ложные результаты. Это общая проблема с удвоениями и imo во многих случаях. Следует избегать Double_s. Может быть, переключение на другой тип - лучшее решение? Например, BigDecimal?
Я ищу способ проверки поля 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.
Вы можете использовать аннотацию, но вы можете получить ложные результаты. Это общая проблема с удвоениями и imo во многих случаях. Следует избегать Double_s. Может быть, переключение на другой тип - лучшее решение? Например, BigDecimal?
Если вы переключились на BigDecimal (или BigInteger), вы можете использовать @DecimalMin или @DecimalMax. Но это все еще не решение для float или double.
Я избегал типов 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 {}
}
Иногда это удобно в паре с @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;
}
}
Вы также можете использовать @Digits
в API-интерфейсе проверки гибернации
@Digits(integer = 10 /*precision*/, fraction = 2 /*scale*/)
Более новые версии Hibernate Validator (по крайней мере 6.0.17) поддерживают аннотацию @DecimalMin/Max
для двойного
Смотрите класс для проверки