我有一个请求WorkerRequest
,其中有一个enum
有FULL_TIME
,MANAGER
等。
在WorkerRequest
中,如何在此枚举上应用长度验证?
示例:枚举类型不应超过8个字符。
FULL_TIME
有效(8个字符)
PERMANENT
无效(9个字符)
目前,如果我把javax.validation.constraints.Size
@Size(min = 0, max = 8, message = "Allowed length for workerType is 8.")
@Enumerated(EnumType.STRING)
private WorkerType workerType;
它会抛出一个错误:
HV000030:找不到约束'javax.validation.constraints.Size'验证类型'com.XX.XX.XX.WorkerType'的验证器。检查'workerType'的配置
如Difference between @Size, @Length and @Column(length=value)所述
@Size是一个Bean Validation批注,用于验证关联的String是否具有长度以最小值和最大值为界的值。
您只能指定db中持久枚举值所需的最大长度。例如,如果您定义@Column(length = 8)
而不是@Size,您将在db定义中相应地看到workerType VARCHAR(8)
。
但它有一个解决方法:假设你有
public enum WorkerType {PERMANENT , FULL_TIME, ...};
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
@Constraint(validatedBy = EnumSizeLimit.class)
public @interface EnumSizeLimit {
String message() default "{com.example.app.EnumSizeLimit.message}";
Class<?>[] groups() default {};
Class<? extends Payload>[] payload() default {};
Class<? extends Enum<?>> targetClassType();
}
public class EnumSizeLimitValidator implements ConstraintValidator < EnumSizeLimit , String > {
private Set < String > allowedValues;
@SuppressWarnings({
"unchecked",
"rawtypes"
})
@Override
public void initialize(EnumSizeLimit targetEnum) {
Class << ? extends Enum > enumSelected = targetEnum.targetClassType();
allowedValues = (Set < String > ) EnumSet.allOf(enumSelected).stream().map(e - > ((Enum << ? extends Enum << ? >> ) e).name())
.collect(Collectors.toSet());
}
@Override
public boolean isValid(String value, ConstraintValidatorContext context) {
return value == null || (value.length>=0 && value.length<=8)) ? true : false;
}
}
@EnumSizeLimit (targetClassType = WorkerType.class, message = "your message"
private String workerType;
我没有看到尺寸验证的需要。在请求中使用枚举的事实将迫使序列化将值映射到枚举可能值之一。如果你用字符串json填充“invalid”作为worker类型,spring将抛出:
org.springframework.http.converter.HttpMessageNotReadableException:JSON解析错误:无法从字符串“invalid”反序列化WorkerType
类型的值:value不是声明的Enum实例名称之一:等等。
您可以在自定义休息异常处理程序中捕获异常