下面来看看Java中注解是如何实现的java
建立注解类Inter:函数
建立测试类Test:测试
在程序第二句设置断点,能够看到:3d
能够看到,注解的实例是一个动态代理类的对象.代理
要想查看这个动态代理类,能够在代码中加调试
System.setProperty("sun.misc.ProxyGenerator.saveGeneratedFiles", "true");
添加系统代理,将其导出为class文件对象
能够看到以下两个文件:blog
反编译$Proxy1.class,以下:继承
能够看到,动态代理类是咱们定义的注解实现类,反编译Inner.class,以下:索引
能够看到,注解接口继承了java.lang.annotation.Annotation, 经过查看源码,该类源码以下:
能够看到, 该类下的方法都被$Proxy1动态代理类实现了.
到此处,咱们已经知道Inner注解(接口)是一个继承了Annotation接口的特殊接口,而咱们经过反射获取注解时,返回的是Java运行时生成的动态代理对象$Proxy1,该类就是Inner注解(接口)的具体实现类。
那么, 代理类是如何处理方法的调用的呢?
咱们知道, 动态代理方法的调用最终会传递给绑定的InvocationHandler实例的invoke方法处理。咱们能够看看$Proxy1的源码
其中语句调用了父类的成员变量,其父类为Proxy, 查看该成员变量,以下:
能够看到, h对象类型就是InvocationHandler接口的某个实现类
咱们在Proxy类的构造方法处设置断点:
经过断点能够查看h具体是哪一个对象:
能够看到, 该动态代理类为AnnotationInvocationHandler对象, 查看该类的invoke方法以下:
其中的memberValues变量是以方法名为key,以变量为value的, 以下:
那么,这个memberValues变量是从哪来的呢?
能够看到,其是在构造函数中进行设置的.
反编译咱们的Test类,看到:
因此中间有一个类,负责建立代理对象AnnotationInvocationHandler, 其将变量从常量池中取出并建立map, 进而建立代理对象, 这个类就是 AnnotationParser, 在此不细说了, 感兴趣的能够自行断点调试查看.
注解本质是一个继承了Annotation的特殊接口,其具体实现类是Java运行时生成的动态代理类。经过代理对象调用自定义注解(接口)的方法,会最终调用AnnotationInvocationHandler的invoke方法。该方法会从memberValues这个Map中索引出对应的值。而memberValues的来源是Java常量池。