Java注解的原理
Java 注解
Java 注解用于为 Java 代码提供元数据。作为元数据,注解不直接影响你的代码执行,但也有一些类型的注解实际上可以用于这一目的。Java 注解是从 Java5 开始添加到 Java 的。
注解其实同class 和Interface 一样,注解也属于一种类型,它是在JavaSE5.0版本中开始引入的概念。注解就相当于标签
注解的定义
注解通过@interface 关键字进行定义
1 | public TestAnnotation{ |
注解的形式和接口很像,只不过前面多了一个@符号,上面的代码就创建了一个名字叫TestAnnotation 的注解。可以简单理解为创建了一张名字为TestAnnotation的标签。
注解的应用
1 |
|
元注解
元注解是可以注解到注解上的注解,或者说元注解是一种基本注解,但是它能够应用到其它的注解上面。
元注解也是一张标签,但是它是一张特殊的标签,它的作用和目的就是给其他普通的标签进行解释说明的。
元标签有 @Retention、@Documented、@Target、@Inherited、@Repeatable 5 种。
- @Retention 保留期
解释说明这个注解的存活时间
RetentionPolicy.SOURCE 注解只在源码阶段保留,在编译器进行编译时它将被丢弃忽视。
RetentionPolicy.CLASS 注解只被保留到编译进行的时候,它并不会被加载到 JVM 中。
RetentionPolicy.RUNTIME 注解可以保留到程序运行的时候,它会被加载进入到 JVM 中,所以在程序运行时可以获取到它们。
- @Documented
它的作用是能够将注解中的元素包含到JavaDoc中去。
- @Target
Target 是目标的意思,@target 指定了注解运用的地方。
ElementType.ANNOTATION_TYPE 可以给一个注解进行注解
ElementType.CONSTRUCTOR 可以给构造方法进行注解
ElementType.FIELD 可以给属性进行注解
ElementType.LOCAL_VARIABLE 可以给局部变量进行注解
ElementType.METHOD 可以给方法进行注解
ElementType.PACKAGE 可以给一个包进行注解
ElementType.PARAMETER 可以给一个方法内的参数进行注解
ElementType.TYPE 可以给一个类型进行注解,比如类、接口、枚举
- @Inherited
继承的意思,是指如果一个超类被@Inherited注解过的注解进行注解的话,那么如果它的子类没有被任何注解应用的话,那么这个子类就继承了超类的注解。
1 |
|
注解 Test 被 @Inherited 修饰,之后类 A 被 Test 注解,类 B 继承 A,类 B 也拥有 Test 这个注解。
- @Repeatable
可重复,就是可以添加多个一样的注解,但是属性可以不一样。
注解的属性
注解的属性也叫做成员变量,注解只有成员变量,没有方法,注解的成员变量在注解中的定义中以“无参的方法”形式来声明,其方法名定义了该成员变量的名字,其返回值定义了该成员变量的类型。
1 | (ElementType.TYPE) |
上面代码定义了TestAnnotation 这个注解拥有id 和msg 两个属性,在使用的时候,应该给他们赋值,赋值的方式是在注解的括号内以value=””形式,多个属性之前用,隔开。
1 | 3,msg="hello annotation") (id= |
注解中的属性可以有默认值,默认值需要用default 关键字指定
1 | (ElementType.TYPE) |
TestAnnotation 中 id 属性默认值为 -1,msg 属性默认值为 Hi。
注解的提取
当写好注解后,现在的工作是检查这些注解,提取上面的信息。
要想正确的检阅注解,需要使用反射。
注解通过反射获取,首先可以通过Class 对象的isAnnotationPresent()方法判断它是否应用了某个注解
1 | public boolean isAnnotationPresent(Class<? extends Annotation> annotationClass) {} |
然后通过getAnnotation()方法来获取Annotation 对象。(返回指定类型的注解)
1 | public <A extends Annotation> A getAnnotation(Class<A> annotationClass) {} |
或者是getAnnotations()方法(返回注解到这个元素上的所有注解)
1 | public Annotation[] getAnnotations() {} |
如果获取到的Annotation 如果不为空,就可以调用他们的属性方法了。比如
1 | () |