前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Java 注解原理

Java 注解原理

作者头像
烟草的香味
发布2019-07-25 10:57:26
6110
发布2019-07-25 10:57:26
举报
文章被收录于专栏:烟草的香味烟草的香味

下面来看看Java中注解是如何实现的

创建注解类Inter:

创建测试类Test:

在程序第二句设置断点,可以看到:

可以看到,注解的实例是一个动态代理类的对象.

要想查看这个动态代理类,可以在代码中加

代码语言:javascript
复制
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常量池。

本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2019-01-30,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 烟草的香味 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 总结
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档