闲话元注解@Retention

本文详细解析Java元注解@Retention的三种级别:SOURCE、CLASS和RUNTIME,通过实例展示不同级别下注解的存在形式及应用,帮助理解注解在编译期和运行期的作用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

元注解@Retention

元注解是用来描述注解的。@Retention用来描述注解将会在哪个层次存在,有三个值:

  1. RetentionPolicy.SOURCE:只会在java文件中存在,class文件中就不可见了。可以被编译器使用
  2. RetentionPolicy.CLASS:会在class中可见,不需要被虚拟机加载。编译时可见
  3. RetentionPolicy.RUNTIME:在class中可见,会被虚拟机加载。编译时可见,运行时可见

下面以一个例子来区分三个值。

Demo

定义一个注解:

@Target(ElementType.FIELD)
@Retention(RetentionPolicy.XXX)
public @interface MyTest {
}

一个类使用了这个注解:

public class MyClass {

    @MyTest
    int val;

}

一个类来获取Field然后获取注解,输出,

try {
            Field field = MyClass.class.getDeclaredField("val");
            Annotation[] annos = field.getAnnotations();
            for (int i = 0; i < annos.length; i++) {
                System.out.println(annos[i].toString());
            }
        } catch (NoSuchFieldException e) {
            e.printStackTrace();
        }

RetentionPolicy.SOURCE

当@Retention注解的值是这个时,编译MyClass得到的class文件如下:

public class MyClass {
    int val;

    public MyClass() {
    }
}

可以看到class文件中没有注解的信息。那这种有什么应用场景呢?

答案是APT(Annotation Process Tool),这个工具的输入是java源文件,因此对于APT而言,是可以看到注解的。
这种情况下,执行程序,输出为空,因为class文件中连注解都没有,肯定是拿不到的。

RetentionPolicy.CLASS

修改为这个后,编译后的class如下:

public class MyClass {
    @MyTest
    int val;

    public MyClass() {
    }
}

可以看到,这个时候,出现了@MyTest注解。但是执行程序,依然输出为空,这是因为java虚拟机加载该类时,没把注解加载进去。

此种的场景使用参考自:

这里很重要的一点是编译多个Java文件时的情况:假如要编译A.java源码文件和B.class文件,其中A类依赖B类,并且B类上有些注解希望让A.java编译时能看到,那么B.class里就必须要持有这些注解信息才行。同时我们可能不需要让它在运行时对反射可见(例如说为了减少运行时元数据的大小之类),所以会选择CLASS而不是RUNTIME。–知乎:RednaxelaFX

RetentionPolicy.RUNTIME

修改为这个后,编译后的class和RetentionPolicy.CLASS一样,执行程序,输出为:

@MyTest()

可以看到,RetentionPolicy.RUNTIME这种情况下是可以通过反射拿到注解的。

参考

关注我的技术公众号,与君共同学习。微信扫一扫下方二维码即可关注:
二维码

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值