30

注解可以有复杂的返回类型,比如HashMap。

我正在寻找类似的东西:

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.FIELD)
public @interface column {
    public HashMap<String, String> table();
}

所以我可以有一个常量注释(伪代码):

@column({table=(dbName, tableName), table=(dbName, tableName2)})
public static final String USER_ID = "userid";

如果 Annotation 不允许您具有复杂的返回类型,那么对于这种情况有什么好的做法吗?

4

2 回答 2

71

不可以,注解元素只能是原始类型、字符串、enum类型Class、其他注解或其中任何一种的数组。表示这些结构的典型方法是声明另一种注释类型

public @interface TableMapping {
  public String dbName();
  public String tableName();
}

然后说

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.FIELD)
public @interface column {
    public TableMapping[] table();
}

并将注释用作

@column(table={
  @TableMapping(dbName="dbName", tableName="tableName"),
  @TableMapping(dbName="db2", tableName="table2")
})
public String userId = "userid";
于 2012-11-26T16:35:45.387 回答
12

几年后,Java 8 为我们带来了。它提供了一种重复同一类注释的方法。

在 Java 8 中,您可以将注解声明为隐式包装在容器注解中。您声明为@Repeated(value=a_class)要可重复的注释。当您添加可重复注释的多个实例时,编译器将自动将这些实例包装在a_class指定为@Repeated.

如果您声明:

@Retention(RetentionPolicy.RUNTIME)
public @interface Columns {
    Column[] value() default {};
}

@Retention(RetentionPolicy.RUNTIME)
@Repeatable( value = Columns.class )
public @interface Column {
    String dbName();
    String tableName();
}

那么您可以多次使用注释,无论是否将它们包装在另一个注释中,两者都是等效的:

@Column(dbName="db1", tableName="table1")
@Column(dbName="db2", tableName="table2")
public static final String USER_ID = "userid";

@Columns({
        @Column(dbName="db3", tableName="table3"),
        @Column(dbName="db4", tableName="table4")
})
public static final String LAST_NAME = "last name";

在这两种情况下都使用检索注释getAnnotationsByType(class)

public static void main(String[] args) {
    for( Field field : AnnotationsTest.class.getDeclaredFields() ){
        System.out.println("Field: " + field.getName());
        Column[] columns = field.getAnnotationsByType(Column.class);
        for( Column column : columns ){
            System.out.println("    db: " + column.dbName() + " table: " + column.tableName());
        }
        System.out.println();
    }
}

它应该输出:

Field: USER_ID
    db: db1 table: table1
    db: db2 table: table2

Field: LAST_NAME
    db: db3 table: table3
    db: db4 table: table4
于 2017-07-30T23:11:01.843 回答