首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >为什么RecordComponent没有在Java17的Records类中定义的注释信息?

为什么RecordComponent没有在Java17的Records类中定义的注释信息?
EN

Stack Overflow用户
提问于 2021-10-13 17:19:54
回答 1查看 107关注 0票数 5

我玩着唱片,发现了一些对我来说不合逻辑的东西:

记录:

代码语言:javascript
复制
record R(@NotEmpty Integer i) {}

代码:

代码语言:javascript
复制
RecordComponent[] recordComponents = R.class.getRecordComponents();
System.out.println(recordComponents[0].isAnnotationPresent(NotEmpty.class));
//prints false

但是,如果我这样做了:

代码语言:javascript
复制
System.out.println(R.class.getDeclaredFields()[0].isAnnotationPresent(NotEmpty.class));
//prints true

这是意料之中的吗?因为RecordComponent implements AnnotatedElement,所以我认为RecordComponent应该有关于注释的信息。我的期望是错误的吗?

EN

回答 1

Stack Overflow用户

回答已采纳

发布于 2021-10-13 18:14:49

这取决于批注的指定允许targets。record组件具有关联的构造函数参数、字段、访问器方法和类型。如果这些位置中至少有一个是允许的,则编译器将接受注释,但该注释将仅与记录的允许位置相关联。

所以下面的代码

代码语言:javascript
复制
public class RecordAnnotations {
    public static void main(String[] args) {
        Class<?> cl = Test.class;
        for(var r: cl.getRecordComponents()) {
            System.out.println("record component "
                + Arrays.toString(r.getAnnotations()) + " " + r);
            System.out.println("\tof type " + r.getAnnotatedType());
            System.out.println("\taccessed by " +
                Arrays.toString(r.getAccessor().getAnnotations())+" "+r.getAccessor());
            System.out.println("\twith return type "
                + r.getAccessor().getAnnotatedReturnType());
        }
        System.out.println();
        for(var r: cl.getDeclaredFields()) {
            System.out.println("field " + Arrays.toString(r.getAnnotations())+" "+r);
            System.out.println("\tof type " + r.getAnnotatedType());
        }
        System.out.println();

        for(var c: cl.getDeclaredConstructors()) {
            System.out.println("constructor " + c);
            for(var r: c.getParameters()) {
                System.out.println("\tparameter "
                    + Arrays.toString(r.getAnnotations()) + " " + r);
                System.out.println("\t\tof type " + r.getAnnotatedType());
            }
        }
        //System.out.println();
        //for(var r: cl.getRecordComponents()) System.out.println(r);
    }
}

@Retention(RetentionPolicy.RUNTIME) @Target(ElementType.RECORD_COMPONENT)
@interface WithRecord {}

@Retention(RetentionPolicy.RUNTIME) @Target(ElementType.FIELD)
@interface WithField {}

@Retention(RetentionPolicy.RUNTIME) @Target(ElementType.PARAMETER)
@interface WithParameter {}

@Retention(RetentionPolicy.RUNTIME) @Target(ElementType.TYPE_USE)
@interface WithTypeUse {}

@Retention(RetentionPolicy.RUNTIME) @Target(ElementType.METHOD)
@interface WithMethod {}

record Test(@WithRecord @WithField @WithParameter @WithTypeUse @WithMethod
            int component) {
}

打印

代码语言:javascript
复制
record component [@WithRecord()] int component
        of type @WithTypeUse() int
        accessed by [@WithMethod()] public int Test.component()
        with return type @WithTypeUse() int

field [@WithField()] private final int Test.component
        of type @WithTypeUse() int

constructor Test(int)
        parameter [@WithParameter()] int component
                of type @WithTypeUse() int

显示如何将每个注释复制到其允许的位置。只有将RECORD_COMPONENT作为允许目标的注释才能通过RecordComponentgetAnnotationisAnnotationPresent进行检索。

当然,@Retention(RetentionPolicy.RUNTIME)也是必需的。

票数 7
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/69559744

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档