元注解是用来描述注解的。@Retention用来描述注解将会在哪个层次存在,有三个值:
下面以一个例子来区分三个值。
定义一个注解:
@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();
}
当@Retention注解的值是这个时,编译MyClass得到的class文件如下:
public class MyClass {
int val;
public MyClass() {
}
}
可以看到class文件中没有注解的信息。那这种有什么应用场景呢?
答案是APT(Annotation Process Tool),这个工具的输入是java源文件,因此对于APT而言,是可以看到注解的。 这种情况下,执行程序,输出为空,因为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
修改为这个后,编译后的class和RetentionPolicy.CLASS一样,执行程序,输出为:
@MyTest()
可以看到,RetentionPolicy.RUNTIME这种情况下是可以通过反射拿到注解的。
本文分享自 每天学点Android知识 微信公众号,前往查看
如有侵权,请联系 cloudcommunity@tencent.com 删除。
本文参与 腾讯云自媒体同步曝光计划 ,欢迎热爱写作的你一起参与!
扫码关注腾讯云开发者
领取腾讯云代金券
Copyright © 2013 - 2025 Tencent Cloud. All Rights Reserved. 腾讯云 版权所有
深圳市腾讯计算机系统有限公司 ICP备案/许可证号:粤B2-20090059 深公网安备号 44030502008569
腾讯云计算(北京)有限责任公司 京ICP证150476号 | 京ICP备11018762号 | 京公网安备号11010802020287
Copyright © 2013 - 2025 Tencent Cloud.
All Rights Reserved. 腾讯云 版权所有