Java 注解原理
下面来看看Java中注解是如何实现的 创建注解类Inter: ![]() 创建测试类Test: ![]() 在程序第二句设置断点,可以看到: ![]() 可以看到,注解的实例是一个动态代理类的对象. 要想查看这个动态代理类,可以在代码中加 System.setProperty("sun.misc.ProxyGenerator.saveGeneratedFiles","true"); 添加系统代理,将其导出为class文件 ![]() 可以看到如下两个文件: ![]() 反编译$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常量池。 (编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |