【Java SE】十六、注解
Java 注解(Annotation)又称 Java 标注,是 JDK5.0 引入的一种注释机制。
Java 语言中的类、方法、变量、参数和包等都可以被标注。和 Javadoc 不同,Java 标注可以通过反射获取标注内容。在编译器生成类文件时,标注可以被嵌入到字节码中。Java 虚拟机可以保留标注内容,在运行时可以获取到标注内容 。 当然它也支持自定义 Java 标注。
Java 定义了一套注解,共有 7 个,3 个在 java.lang 中,剩下 4 个在 java.lang.annotation 中。
作用在代码的注解:
- @Override:检查该方法是否是重写方法。如果发现其父类,或者是引用的接口中并没有该方法时,会报编译错误。
- @Deprecated:标记过时方法。如果使用该方法,会报编译警告。
- @SuppressWarnings:指示编译器去忽略注解中声明的警告。
作用在其他注解的注解(或者说 元注解):
- @Retention:标识这个注解的生命周期,是只在源代码中,还是编入 class 文件中,或者是在运行时可以通过反射访问。
- @Documented:标记这些注解是否包含在 Javadoc 中。
- @Target:标记这个注解可以修饰哪种 Java 成员。
- @Inherited:标记这个注解是可被子类继承的。
从 Java 7 开始,额外添加了 3 个注解:
- @SafeVarargs:Java 7 开始支持,忽略任何使用参数为泛型变量的方法或构造函数调用产生的警告。
- @FunctionalInterface:Java 8 开始支持,标识一个匿名函数或函数式接口。
- @Repeatable:Java 8 开始支持,标识某注解可以在同一个声明上使用多次。
基本注解
这三个注解常用在代码当中,如下所示:
interface Info {
void fuck();
}
class Shit implements Info {
@Override // 检查是否正确重写方法
void fuck() {
System.out.println("Oh,fuck!");
}
@Deprecated // 表明方法已过时,不建议使用
void shit() {
@SuppressWarnings("unused") // 忽略变量未使用的警告
String desire = "breast";
System.out.println("Oh,shit!");
}
}
自定义注解
在 Java 中自定义注解需要用到 @interface
修饰符,但它跟接口没啥关系。
public @interface MyAnnotation {
String value(); // 单参数形式,也可以无参数,跟@Override一样
// String value() default "fuck"; - 还可以指定默认参数
// String[] value() - 如果要传入多个值,可以像这样定义
}
/* 跟据自定义的类型,选择合适的形式
* @MyAnnotation 无参或默认参数形式,作为标识
* @MyAnnotation("hello") 传参形式
* @MyAnnotation({"fuck", "your", "mother"}) 多参形式
*/
class Test {...}
自定义注解必须配上注解的信息处理流程(使用反射)才有意义,反射部分后续再讲。
元注解
元注解就是修饰注解的注解,Java 提供了四种元注解,如上文所述,示例如下:
@Retention(RetentionPolicy.RUNTIME) // 标识该注解可在运行时访问
@Target({TYPE, FIELD, METHOD, PARAMETER, CONSTRUCTOR, LOCAL_VARIABLE}) // 标识该注解可标注的Java成员
@Documented // 标识该注解可包含在Javadoc中
@Inherited // 标识该注解可被所修饰的父类的子类继承
public @interface MyAnnotation {String value();}
可重复注解
这是 JDK 8 新增的一个新特性,示例如下:
@interface MyAnnotations {MyAnnotation[] value();} // 先定义复数类
@Repeatable(MyAnnotations.class) // 在声明这个
public @interface MyAnnotation {String value();}
@MyAnnotation("hello")
@MyAnnotation("world") // 重复注解
class Test {...}
注意:复数类必须与原来的注解类设定相同,如:生命周期、可修饰类型等
类型注解
JDK 8 中 @Target
新增了两个修饰类型 TYPE_PARAMETER
, TYPE_USE
,使得注解可以应用在更多的地方,示例如下:
class Generic<@MyAnnotation T> { // 需要在@Target中添加TYPE_PARAMETER
public void show() throws @MyAnnotation RuntimeException { // 需要在@Target中添加TYPE_USE
ArrayList<@MyAnnotation String> list = new ArrayList<>(); // 需要在@Target中添加TYPE_USE
int num = (@MyAnnotation int) 10L; // 需要在@Target中添加TYPE_USE
}
}