Java注释:将字段值用于另一个字段

时间:2017-01-09 17:00:00

标签: java java-annotations

我有以下注释类

public @interface Size {
  int min() default 1;
  int max() default 100;
  String message() default "Age between min - max";
}

此处默认message()我想要默认值min()max()。简单地写String message() default "Age between" + min() + "-" + max();在这里不起作用。有没有直接的方法呢?

编辑1: 我也有一个人课

public class Person {

  @Size(max = 10)
  private String name;

  @Size(min = 18, message = "Age can not be less than {min}")
  private int age;

  public Person(String s, int i) {
    this.name = s;
    this.age = i;
  }
}

现在,可以设置min()max()值。因此,如果用户输入错误,则message()将相应地打印机。

编辑2: 正如@nicolas想要的那样。这里是验证输入和打印错误消息的AnnonatedValidator类。

public class AnnotatedValidator {

public static void validate(Person p, List<ValidationError> errors) {

    try {
        Field[] fields = p.getClass().getDeclaredFields();
        for(Field field : fields) {
            if(field.getType().equals(String.class)){
                field.setAccessible(true);
                String string = (String)field.get(p);

                Annotation[] annotationsName = field.getDeclaredAnnotations();
                for (Annotation annotation : annotationsName){
                    if (annotation instanceof Size){
                        Size size = (Size) annotation;
                        if (string.length() < size.min() || string.length() > size.max()) {
                            error(size, errors);
                        }
                    }
                }

            } else if (field.getType().equals(int.class)) {
                field.setAccessible(true);
                int integer = (Integer)field.get(p);

                Annotation[] annotationsAge = field.getDeclaredAnnotations();
                for (Annotation annotation : annotationsAge){
                    if (annotation instanceof Size){
                        Size size = (Size) annotation;
                        if (integer < size.min() || integer > size.max()) {
                            error(size,errors);
                        }
                    }
                }
            }
        }

    }catch (IllegalAccessException e) {
        e.printStackTrace();
    }
}

public static void print(List<ValidationError> errors) {
    for (int i = 0; i < errors.size(); i++){
        System.out.println("Errors: " + errors.get(i).getError());
    }
}

public static void error (Size size, List<ValidationError> errors) {
    String error = size.message();
    if (!error.equals(null)) {
        if (error.contains("min")) {
            error = error.replace("min", ""+size.min());
        }
        if (error.contains("max")){
            error = error.replace("max", ""+size.max());
        }
    }

    ValidationError v = new ValidationError();
    v.setError(error);
    errors.add(v);
}
}

ValidationError是另一个只保存错误的类。

1 个答案:

答案 0 :(得分:3)

不,default类型的String值必须是常量表达式,如Java Language Specification dictates

  

如果元素类型不相称,则为编译时错误   与元素值。元素类型T与a相称   元素值V当且仅当满足下列条件之一时:

     
      
  • [...]
  •   
  • T不是数组类型,V的类型是赋值兼容的   (§5.2)与T,并且:      
        
    • [...]
    •   
    • 如果T是基本类型或String,则V是常量表达式(第15.28节)。
    •   
  •   

另一个注释元素的调用不是常量表达式。

您需要在管理注释使用的组件中处理此问题。将default消息声明为某个特殊值。

String message() default "REPLACE_ME";

然后在构造消息时检查它。例如

Field field = ... // get 'age' field
Size size = field.getAnnotation(Size.class);
if (size != null) {
    String message = size.message();
    if (message.equals("REPLACE_ME")) {
        message = "Age between " + size.min() + " - " + size.max() + "."; 
    }
}
int min = size.min();
int max = size.max();
// if field is of type int
int value = field.getInt(instance);
if (value > max || value < min) {
    throw new ConstraintViolationException(message);
}