我有一个这样的实体:
@Entity
@Table(name = "transaction_receiver")
public class TransactionReceiver implements Serializable, Addressable {
private static final long serialVersionUID = 1L;
@Id
@GeneratedValue(strategy = GenerationType.AUTO)
private Long id;
@NotNull
@Column(name = "contact_id", nullable = false)
private String contactId;
@Column(name = "datev_number")
private String datevNumber;
@NotNull
@Column(name = "receiver", nullable = false)
private String receiver;
@NotNull
@Size(min = 22, max = 34)
@Column(name = "iban", length = 34, nullable = false)
private String iban;
@Size(min = 8, max = 11)
@Column(name = "bic", length = 11, nullable = false)
private String bic;
@NotNull
@Column(name = "institute")
private String institute;
@Column(name = "company")
private String company;
和任务,进行自定义验证“你可以提供一个空的iban,bic和研究所,没关系。但是如果任何一个字段不为空,那么上面的constrait必须持有”
我正在寻找最优雅的方式来实现这一目标。
我目前的解决方案是 - 我认为有点脏,但有效 - 使用@PrePersist语句并从那里抛出异常
@PrePersist
public void checkBankData() {
boolean ibanEmpty = iban == null || iban.isEmpty();
boolean ibanValid = !ibanEmpty && iban.length() >= 22 && iban.length() <= 34;
boolean bicEmpty = bic == null || bic.isEmpty();
boolean bicValid = !bicEmpty && bic.length() >= 8 && bic.length() <= 11;
boolean instituteEmpty = institute == null || institute.isEmpty();
boolean validState = (ibanEmpty && bicEmpty && instituteEmpty) || ibanValid && bicValid;
if (!validState) {
throw new IllegalStateException(
String.format(
"bank data is not empty and %s%s%s%s%s",
!ibanValid ? "iban has to be from 22 to 34 chars long" : "",
!ibanValid && !bicValid ? "and" : "",
!bicValid ? "bic has to be from 8 to 11 chars long" : "",
!ibanValid && !bicValid && instituteEmpty ? "and" : "",
instituteEmpty ? "institue must not be empty" : ""
)
);
}
}
对@Valid注释不明智。另一种方法是定义如下所述的自定义验证器:http://docs.jboss.org/hibernate/validator/4.1/reference/en-US/html/validator-customconstraints.html
但这对我的约束来说真的有点过分了。
有没有其他优雅的方式?
由于
答案 0 :(得分:5)
使用Hibernate验证API并不像看起来那么复杂,并且对于你的约束来说是一个很好的解决方案。 但是,您可以使用Hibernate Validator更简单地定义约束,就像我们在一个添加了几个类的项目中所做的那样。您的约束将如下所示:
@Validate(method = "checkBankData", message = "{BankData.invalid.message}")
@Entity
@Table(name = "transaction_receiver")
public class TransactionReceiver implements Serializable, Addressable {
要实现此目的,您需要定义@Validate注释和CustomValidator类。
@Target({ ElementType.TYPE, ANNOTATION_TYPE })
@Retention(RUNTIME)
@Constraint(validatedBy = CustomValidator.class)
@Documented
/**
* Annotation to allow custom validation against model classes
*/
public @interface Validate {
/**
* Validation message
*/
String message();
Class<?>[] groups() default {};
Class<? extends Payload>[] payload() default {};
/**
* Validation method name
*/
String method() default "";
}
public class CustomValidator implements ConstraintValidator<Validate, BusinessObject> {
private static Log log = LogFactory.getLog(CustomValidator.class);
private String validator;
@Override
public void initialize(Validate constraintAnnotation) {
validator = constraintAnnotation.method();
}
@Override
public boolean isValid(BusinessObject bo, ConstraintValidatorContext constraintContext) {
try {
return isValidForMethod(bo);
} catch (Exception e) {
/* Error durante la ejecución de la condición o del validador */
log.error("Error validating "+bo, e);
return false;
}
}
private boolean isValidForMethod(BusinessObject bo) throws Exception {
Method validatorMethod = ReflectionUtils.findMethod(bo.getClass(), validator, new Class[] {});
if (validatorMethod != null) {
/* Validator call */
Boolean valid = (Boolean) validatorMethod.invoke(bo);
return valid != null && valid;
} else {
/* Method not found */
log.error("Validator method not found.");
return false;
}
}
}
如果您计划定义更多约束,这种方法会很好。您可以使用更多功能扩展它,例如验证条件或添加多个验证等等。
题外话:
验证与Spring Boot无关,因此无需在您的问题中提及。
serialVersionUID = 1L;这是一个非常糟糕的主意。使用IDE serialVersionUID生成器为此字段填充1L不同的值。
答案 1 :(得分:0)
要绕过默认的JPA验证而不是Hibernate验证器的@NotEmpty,您可以尝试使用@Size注释来指定它必须大于或小于某个数字。
@NotNull
@Size(min=1)
private String description;
这是我能想到的最清洁的方式。