一、注解概念
注解: 说明程序的,给计算机看的
注释: 用文字描述程序,给程序员看的
1.1、概念
注解(Annotation),也叫元数据。一种代码级别的说明。它是JDK1.5及以后版本引入的一个特性,与类、接口、枚举是在同一个层次。它可以声明在包、类、字段、方法、局部变量、方法参数等的前面,用来对这些元素进行说明,注释。也可以把注解理解为代码里的特殊标记,这些标记可以在编译,类加载,运行时被读取,并执行相应的处理。通过注解开发人员可以在不改变原有代码和逻辑的情况下在源代码中嵌入补充信息。
1.2、作用分类
- 编写文档:通过代码里标识的注解生成文档【生成文档doc文档】
- 代码分析:通过代码里标识的注解对代码进行分析【使用反射】
- 编译检查:通过代码里标识的注解让编译器能够实现基本的编译检查【Override】
二、JDK内置注解
Java从1.5版本以后默认内置三个标注:
- @Override:检查被该注解标注的方式是否是继承自父类【接口】
- @Deprecated: 该注解表示注释的内容过时
- @SuppressWarnings: 压制警告,一般传递参数all 【@SuppressWarnings(“all”)】
@SuppressWarnings压制警告参数值:
all to suppress all warnings (抑制所有警告)
boxing to suppress warnings relative to boxing/unboxing operations(抑制装箱、拆箱操作时候的警告)
cast to suppress warnings relative to cast operations (抑制映射相关的警告)
dep-ann to suppress warnings relative to deprecated annotation(抑制启用注释的警告)
deprecation to suppress warnings relative to deprecation(抑制过期方法警告)
fallthrough to suppress warnings relative to missing breaks in switch statements(抑制确在switch中缺失breaks的警告)
finally to suppress warnings relative to finally block that don’t return (抑制finally模块没有返回的警告)
hiding to suppress warnings relative to locals that hide variable()
incomplete-switch to suppress warnings relative to missing entries in a switch statement (enum case)(忽略没有完整的switch语句)
nls to suppress warnings relative to non-nls string literals(忽略非nls格式的字符)
null to suppress warnings relative to null analysis(忽略对null的操作)
rawtypes to suppress warnings relative to un-specific types when using generics on class params(使用generics时忽略没有指定相应的类型)
restriction to suppress warnings relative to usage of discouraged or forbidden references
serial to suppress warnings relative to missing serialVersionUID field for a serializable class(忽略在serializable类中没有声明serialVersionUID变量)
static-access to suppress warnings relative to incorrect static access(抑制不正确的静态访问方式警告)
synthetic-access to suppress warnings relative to unoptimized access from inner classes(抑制子类没有按最优方法访问内部类的警告)
unchecked to suppress warnings relative to unchecked operations(抑制没有进行类型检查操作的警告)
unqualified-field-access to suppress warnings relative to field access unqualified (抑制没有权限访问的域的警告)
unused to suppress warnings relative to unused code (抑制没被使用过的代码的警告)
三、自定义注解
3.1、注解的本质
格式:
// 元注解
public @interface 注解名称{
// 属性列表
}
自定义的注解反编译后的内容
public interface MyAnno extends java.lang.annotation.Annotation{
}
注解的本质其实就是一个接口,继承Annotation父接口
3.2、注解的返回类型
注解的属性:等同于在接口中定义的抽象方法
返回结果必须是如下类型:
- 基本数据类型
- String类型
- 枚举类型
- 注解
- 以上类型的数组
3.3、属性赋值注意点
- 如果定义的属性时,使用default关键字给属性默认初始值,可以在使用注解是不赋值。
- 如果只有一个属性需要赋值,而且该属性的名称是
value
,那么在赋值时value
可以省略。 - 数组赋值的时候,值使用
{}
包裹,如果数组中只有一个值,那么{}
可以省略。
四、元注解
JDK中给我们提供的4个元注解
注解 | 说明 |
---|---|
@Target | 定义注解的作用目标 |
@Retention | 定义注解的保留策略。RetentionPolicy.SOURCE:注解仅存在于源码中,在class字节码文件中不包含;RetentionPolicy.CLASS:默认的保留策略,注解会在class字节码文件中存在,但运行时无法获得;RetentionPolicy.RUNTIME:注解会在class字节码文件中存在,在运行时可以通过反射获取到。 |
@Document | 说明该注解将被包含在javadoc中 |
@Inherited | 说明子类可以继承父类中的该注解 |
@Repeatable | 表示其(meta-)注释声明的注释类型是可重复的 。(@since 1.8) |
@Native | 表示定义常量值的字段可以从本地代码引用。(@since 1.8) |
@Target类型说明
Target类型 | 说明 |
---|---|
ElementType.TYPE | 类,接口(包括注释类型)或枚举声明 |
ElementType.FIELD | 字段声明(包括枚举常数) |
ElementType.METHOD | 方法声明 |
ElementType.PARAMETER | 正式参数声明 |
ElementType.CONSTRUCTOR | 构造函数声明 |
ElementType.LOCAL_VARIABLE | 局部变量声明 |
ElementType.ANNOTATION_TYPE | 注解类型声明 |
ElementType.PACKAGE | 包装声明 |
ElementType.TYPE_PARAMETER | 键入参数声明 (@since 1.8) |
ElementType.TYPE_USE | 使用类型( @since 1.8) |
@Target({ElementType.TYPE,ElementType.METHOD,ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
public @interface MyAnno {
}
五、自定义注解的案例
/**
* 自定义注解
* 该注解表面要执行哪个类中的哪个方法
*/
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface InvokAnno {
String className();
String methodName();
}
public class Student1 {
public void show(){
System.out.println("student1 show ....");
}
}
@InvokAnno(className = "com.edu.anno2.Student1",methodName = "show")
public class MyMain {
public static void main(String[] args) throws Exception {
// 获取类对象
Class<MyMain> clazz = MyMain.class;
// 获取类对象中的注解
InvokAnno an = clazz.getAnnotation(InvokAnno.class);
/**
* 注解本质是 接口 获取到的其实是接口的实现
* public class MyInvokAnno implements InvokAnno{
*
* String className(){
* return "com.edu.anno2.Student1";
* }
* String methodName(){
* return "show";
* }
* }
*/
// 获取注解中对应的属性
String className = an.className();
String methodName = an.methodName();
System.out.println(className);
System.out.println(methodName);
// 通过反射的方式实现接口的功能
Class<?> aClass = Class.forName(className);
Method show = aClass.getDeclaredMethod("show");
// 方法的执行
Object o = aClass.newInstance();
show.invoke(o); // 执行对应的方法
}
}