Java注释可以具有复杂的返回类型,如HashMap吗?

2022-09-01 10:12:21

可以具有复杂返回类型的注释,例如HashMap。

我正在寻找类似的东西:

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

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

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

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


答案 1

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

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";

答案 2

几年后,我们带来了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