假设我们具有以下结构:
public class User {
@NotNull(message = "not available")
private String phoneNumber;
//getter and setter
}
public class CEO extends User{}
public class TechManager extends User{}
public class Company {
@Valid
private CEO ceo;
@Valid
private TechManager techManager;
//getters and setters
}
在上面的示例中,phoneNumber是CEO和TechManager之间的共享对象。
如何在CEO具有可选功能且TechManager具有必需的phoneNumber的情况下对此进行验证?
答案 0 :(得分:0)
您可以在子类中隐藏private String phoneNumber
字段,并按照每个子类所需的方式定义验证注释。在这种情况下,应该覆盖getter和setter。
答案 1 :(得分:0)
根据Bean Validation Specification 1.0和Bean Validation 2.0 Specification,您都可以使用验证组。
您可以创建自己的组层次结构。例如,每个字段一组,然后收集一个角色:
首先,声明要为其配置验证的组。
// Define group for which phoneNumber will be mandatory
interface PhoneNumberIsMandatory {}
// Define group for which Email will be mandatory
interface EmailIsMandatory {}
// Define a TechManagerValidation group with includes PhoneNumberIsMandatory and EmailIsMandatory groups
interface TechManagerValidation extends PhoneNumberIsMandatory, EmailIsMandatory {}
让我们准备父类声明。您可以在一组中声明一个或多个字段。另外,您可以为一个字段声明一个或多个组。还是根本不声明组。
class User {
@NotNull(message = "Name nust be set")
public String name;
@NotNull(groups = {PhoneNumberIsMandatory.class}, message = "Phone number must be set")
public String phoneNumber;
@Email
@NotNull(groups = {EmailIsMandatory.class}, message = "Primary email must be set")
String primaryEmail;
@Email
@NotNull(groups = {EmailIsMandatory.class}, message = "Secondary email must be not null")
String secondaryEmail;
}
使用@GroupSequence批注重新定义默认验证组。
@GroupSequence({TechManager.class, TechManagerValidation.class})
class TechManager extends User {
}
现在您可以定义公司类别并检查验证的工作原理
class CEO extends User {
}
class Company {
@Valid
public CEO ceo;
@Valid
public TechManager techManager;
}
public class DemoApplication {
public static void main(String[] args) {
ValidatorFactory factory = Validation.buildDefaultValidatorFactory();
Validator validator = factory.getValidator();
Company company = new Company();
company.ceo = new CEO();
company.techManager = new TechManager();
System.out.println("Company: " + validator.validate(company));
}
}
请注意,“默认”组首先约束已验证,如果其中任何一个失败,则将跳过TechManagerValidation。如果需要,我们可以通过为User类的所有字段定义组来获取所有约束:
interface NameIsMandatory {}
interface PhoneNumberIsMandatory {}
interface EmailIsMandatory {}
interface TechManagerValidation extends NameIsMandatory, PhoneNumberIsMandatory, EmailIsMandatory {}
interface CeoValidation extends NameIsMandatory {}
class User {
@NotNull(groups = {NameIsMandatory.class}, message = "Name nust be set")
public String name;
@NotNull(groups = {PhoneNumberIsMandatory.class}, message = "Phone number must be set")
public String phoneNumber;
@Email
@NotNull(groups = {EmailIsMandatory.class}, message = "Primary email must be set")
String primaryEmail;
@Email
@NotNull(groups = {EmailIsMandatory.class}, message = "Secondary email must be not null")
String secondaryEmail;
}
@GroupSequence({TechManager.class, TechManagerValidation.class})
class TechManager extends User {
}
@GroupSequence({CEO.class, CeoValidation.class})
class CEO extends User {
}
class Company {
@Valid
public CEO ceo;
@Valid
public TechManager techManager;
}
public class DemoApplication {
public static void main(String[] args) {
ValidatorFactory factory = Validation.buildDefaultValidatorFactory();
Validator validator = factory.getValidator();
Company company = new Company();
company.ceo = new CEO();
company.techManager = new TechManager();
System.out.println("Company: " + validator.validate(company));
}
}
还有另一种方法,但我认为它破坏了封装。您可以删除NameIsMandatory,PhoneIsMandatory和EmailIsMandatory组,并定义User类约束,如下所示:
class User {
@NotNull(groups = {TechManagerValidation.class, CeoValidation.class}, message = "Name nust be set")
public String name;
@NotNull(groups = {TechManagerValidation.class}, message = "Phone number must be set")
public String phoneNumber;
@Email
@NotNull(groups = {TechManagerValidation.class}, message = "Primary email must be set")
String primaryEmail;
@Email
@NotNull(groups = {TechManagerValidation.class}, message = "Secondary email must be not null")
String secondaryEmail;
}