最近在看java的注解的源码,要了解注解的运行原理,需要查阅rt.jar 的 java.lang.annotation、sun.reflect.annotation 这两个package的源码,还有了解java的proxy机制。
所有的注解都是继承于 Annotation 接口的,但 @interface 定义的类又没有具体的实现,所以可以推测jvm在实际运行的时候,应该会给@interface 生成一个代理实现类。这个等下在调试的时候可以看到。
1、首先来一段注解的demo
注解类:
package test.annotation;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
public @interface DemoAnnotation {
String name() default "";
int value() default 0;
}
注解测试类:
package test.annotation;
@DemoAnnotation(name = "demo value", value = 1)
public class DemoMain {
public static void main(String[] args) {
DemoAnnotation demoAnnotation = DemoMain.class.getAnnotation(DemoAnnotation.class);
if (demoAnnotation != null) {
String name = demoAnnotation.name();
int value = demoAnnotation.value();
System.out.println("demo annotation name:" + name
+ " value:" + value);
}
}
}
首先看注解的 @Retention 和 @Target
Retention指定注解的保留阶段,目前有三种阶段:SOURCE、CLASS、RUNTIME;
Target 指定注解的作用范围,定义的范围有:TYPE、FIELD、METHOD、PARAMETER、CONSTRUCTOR、LOCAL_VARIABLE、ANNOTATION_TYPE、PACKAGE、TYPE_PARAMETER、TYPE_USE;
然后看DemoMain类,其中有 DemoAnnotation demoAnnotation = DemoMain.class.getAnnotation(DemoAnnotation.class); 这里就获取到了一个注解的代理类,见下图:
既然是代理类,那就有InvocationHandler类实现方法调用的具体实现,由调试可以看到有 AnnotationInvocationHandler 类,该类在 sun.reflect.annotation 中,可以通过查看该类的源码看出有一个map保存了注解方法名的只,其中key为方法名,value为注解使用的时候设置的值。
对于代理类如何生成,还可以继续追查下去,分析一下TestMain的字节码,这样就可以理解整个注解的运行机制了。
以上只是个人的浅析见解,如有不妥之处,请大家多多指点!