文章目录
注解引入始于Java5,用来描述Java代码的元信息。
元注解
元注解的作用是负责注解其他注解。 Java5.0定义了4个标准的meta-annotation(元注解)类型,它们被用来提供对其它注解类型进行说明。
@Target
注解的作用目标,可以在哪些地方使用该注解。
注解的属性
属性名称 | 属性类型 | 属性默认值 | 属性描述 |
---|---|---|---|
value | ElementType[] | 注解的作用目标元素类型枚举集合 |
ElementType枚举的取值,可参考文章:Java 注解的使用范围ElementType
@Retention
注解的保留策略,注解何种情况下仍然存在。
注解的属性
属性名称 | 属性类型 | 属性默认值 | 属性描述 |
---|---|---|---|
value | RetentionPolicy | 注解的保留策略 |
RetentionPolicy枚举的取值
- SOURCE:源文件中存在,编译后的Class文件不存在
- CLASS:源文件、Class文件中存在,运行时不存在
- RUNTIME:源文件、Class文件、运行时都存在,所以可以在运行中通过反射获取到注解信息
@Documented
将此注解包含在javadoc中 。它代表着此注解会被javadoc工具提取成文档。
@Inherited
允许子类继承父类中的注解。如果一个使用了@Inherited 修饰的注解被用于一个类,则这个注解也将被用于该类的子类。
内置注解
@Override
指明被注解修饰的方法是覆写父类中的方法。
如果某个方法使用了该注解,却没有覆写父类中的方法,编译器就会生成一个错误。
在子类中覆写父类或接口的方法的过程中,@Overide其实并不是必须的。但是还是建议使用这个注解。如果加了@Overide注解,在你没有正确覆写父类方法的情况下,编译器会有错误提示信息。
覆写失败的情况举例
- 子类方法的大小写写错了、或者参数错了、或者是子类自己定义的方法;
- 修改了父类的方法的名字,那么之前重写的子类方法将不再属于重写。
@Deprecated
指明被注解修饰的类、方法、变量等不建议使用。
当编译器遇到了使用@Deprecated修饰的类、方法、变量等时会提示相应的警告信息。
@Suppresswarnings
关闭编译器生成的警告信息。可以修饰类、方法、变量等。
它可以达到抑制编译器编译时产生警告的目的,使用@SuppressWarnings注解,采用就近原则,比如一个方法出现警告,尽量使用@SuppressWarnings注解这个方法,而不是注解方法所在的类。所属范围越小越好,因为范围大了,不利于发现该类下其他方法的警告信息。
但是不太建议使用@SuppressWarnings注解,使用此注解,开发人员看不到编译时编译器提示的相应的警告,不能选择更好、更新的类、方法或者不能编写更规范的编码。同时后期更新JDK、jar包等源码时,使用@SuppressWarnings注解的代码可能受新的JDK、jar包代码的支持,出现错误,仍然需要修改。
注解处理器
注解处理器(Annotation Processor Tool)简称APT,它是用于注解处理的命令行程序,它可以找到源码中对应注解的对象并使用注解处理器对其进行处理。
Java中有默认的注解处理器,使用者也可以自定义注解处理器,注册后使用注解处理器处理注解,最终达到注解本身起到的效果。
注解处理器的原理就是:在需要使用的元素上(类、变量、方法、参数等)加上注解,然后在编译时把使用了这个注解的元素都收集起来,做统一的处理。不管是运行时注解还是编译时注解,都会通过处理器在编译时进行扫描和处理注解。
注解处理器将标记了注解的类,变量等作为输入内容,经过注解处理器处理,生成想要生成的java代码。所以处理器可以理解为就是一个生成代码的工具,只是是通过注解的规则生成。生成后的代码,可以看作是同一般代码,最终被编译。
详细了解注解处理器可参考文章:https://blog.csdn.net/u013045971/article/details/53509237
读取注解
读取注解的核心是java.lang.reflect.AnnotatedElement接口。该接口用于获取类、方法、字段等的注解信息。
实现了这个接口的实现类的实例对象代表了在当前JVM中的一个“被注解元素”。使用这个接口中声明的方法可以读取(通过Java的反射机制)“被注解元素”的注解。
这个接口中的所有方法返回的注解都是不可变的、并且都是可序列化的。因此这个接口中所有方法返
回的数组可以被调用者修改,而不会影响其返回给其他调用者的数组。
接口的常用方法
-
default boolean isAnnotationPresent(Class<? extends Annotation> annotationClass)
判断该元素是否存在指定类型的注解,存在返回true,否则返回false。注意:此方法会忽略注解对应的注解容器。
-
<T extends Annotation> T getAnnotation(Class<T> annotationClass)
返回该元素上存在的指定类型的注解,如果该类型注解不存在,则返回null。
-
Annotation[] getAnnotations()
返回该元素上存在的所有注解,若没有注解,返回空数组。
-
default <T extends Annotation> T[] getAnnotationsByType(Class\<T> annotationClass)
返回元素上存在的指定类型的注解数组,没有注解对应类型的注解时,返回空数组。
该方法的调用者可以随意修改返回的数组,而不会对其他调用者返回的数组产生任何影响。
该方法与getAnnotation的区别在于:getAnnotationsByType会检测注解对应的重复注解容器。若该元素为类,当前类上找不到注解,且该注解为可继承的,则会去父类上检测对应的注解。
-
default <T extends Annotation> T getDeclaredAnnotation(Class\<T> annotationClass)
返回该元素上直接存在的指定类型的注解,如果该类型注解不存在,则返回null。
该方法与此接口中的其他方法有所不同,该方法会忽略继承的注解。如果注解没有直接存在于此元素上,而是存在于元素的父类上,也会返回null。
-
default <T extends Annotation> T[] getDeclaredAnnotationsByType(Class\<T> annotationClass)
返回元素上存在的指定类型的注解数组,没有注解对应类型的注解时,返回空数组。该方法会忽略继承的注解。
-
Annotation[] getDeclaredAnnotations()
返回该元素上存在的所有注解,若没有注解,返回空数组。该方法将忽略继承的注解。
接口的部分实现类
- AccessibleObject:可访问对象,如:方法、构造器、属性等
- Class:类
- Constructor:构造器
- Executable:可执行的,如:构造器、方法
- Field:属性
- Method:方法
- Package:包
- Parameter:参数
接口的部分子接口
- AnnotatedArrayType :被注解的数组类型
- AnnotatedParameterizedType :被注解的参数化类型
- AnnotatedType :被注解的类型
- AnnotatedTypeVariable :被注解的类型变量
- AnnotatedWildcardType :被注解的通配类型
- GenericDeclaration :通用声明,用于表示声明型元素,如:类、方法、构造器等
- TypeVariable<D> :类型变量