注解(Annotation),也叫元数据。一种代码级别的说明。JDK1.5之后引入的一个特性,与类、接口、枚举是在同一个层次。它可以声明在包、类、字段、方法、局部变量、方法参数等的前面,用来对这些元素进行说明,注释。
jdk内置的一些注解
名称作用@Override检测被该注解标注的方法是否是继承自父类(接口)的@Deprecated该注解标注的内容,表示已过时@SuppressWarnings压制警告传递参数all @SuppressWarnings("all"),表示压制所有警告@SafeVarargs断言声明的构造函数和方法的主体不会对其varargs参数执行潜在的不安全的操作@FunctionalInterface用于接口上,表明该接口是一个函数式接口对下面自定义注解的class文件反编译
public @interface MyAnno { String show1(); // String类型 int show2(); // 基本数据类型 MyAnno2 an2(); // 自定义的注解类型 Season sea(); // 枚举类型 }反编译后可发现,MyAnno继承了java.lang.annotation.Annotation接口。所以注解本质上是一个接口,
//字节码反编译后 public abstract interface MyAnno implements Annotation { <ClassVersion=52> <SourceFile=MyAnno.java> public abstract show1() {} //()Ljava/lang/String; public abstract show2() {} //()I public abstract an2() {} //()Lanno/MyAnno2; public abstract sea() {} //()Lanno/Season; 返回类型--枚举类型 }就是接口中的抽象方法
属性的返回值类型 基本数据类型,String,枚举,注解,以上类型的一维数组 定义了属性,使用时需要给属性赋值 使用default关键字给属性默认初始化值,使用注解时,可以不进行属性的赋值。如果只有一个属性需要赋值,并且属性的名称是value,则value可以省略,直接定义值即可。数组赋值时,值使用{}包裹。如果数组中只有一个值,则{}可以省略自定义注解示例:
注解内属性允许的属性 public @interface MyAnno { String show1(); // String类型 int show2(); // 基本数据类型 MyAnno2 an2(); // 注解类型 Season sea(); // 枚举类型 } // 枚举类 public enum Season { Spring,Summer,Winter,Auto; } 带默认值的注解属性 public @interface MyAnno3 { String name(); int age() default 12; // 带默认值的属性 } 注解中只有一个属性,并且名称为value public @interface MyAnno4 { String value(); }使用示例:
@SuppressWarnings("all") // 忽略所有警告 public class Demo1 { @Override // 表示对父类方法的重写 public String toString() { return "Demo1{}"; } @Deprecated // 表明该方法已过时 public void show1(){ // ... } @SuppressWarnings("all") // 忽略该方法的警告 public void show2(){ //新方法,替代show1 } @MyAnno3(name = "Yau") public void demo1(){ //调用show1,显示删除线 show1(); } @MyAnno4("hhhh") public void testAnno4(){ //如果注解里面只要一个属性,并且名字叫valua,那么可以省略不写 } @MyAnno(show1 = "Yangg",show2 = 3,an2 = @MyAnno2,sea = Season.Spring) public void testAnno(){ } }概念:元注解是用于描述注解的注解
名称作用@Target表示该注解能够作用的范围,它的取值有: ElementType.TYPE:作用于类上 ElementType.METHOD:作用于方法上 ElementType.FIELD:作用于成员变量上上面是比较常用的几个@Retention表示该注解被保留的阶段,它的取值有:RetentionPolicy.RUNTIME:注解会保留到class字节码文件中,并被JVM读取到RetentionPolicy.CLASS:注解保留到编译期RetentionPolicy.SOURCE 注解只在源码阶段保留@Documented表示该该注解是否被抽取到api文档中@Inherited表示该注解是否被子类继承使用示例:
/** 元注解:用于描述注解的注解 -- @Target:描述注解能够作用的位置 -- @Retention:描述注解被保留的阶段 -- @Documented:描述注解是否被抽取到api文档中 -- @Inherited:描述注解是否被子类继承 */ //ElementType.TYPE:表示该注解只能作用于类上 @Target(value = {ElementType.TYPE,ElementType.METHOD,ElementType.FIELD}) @Retention(value = RetentionPolicy.RUNTIME) public @interface YuanAnno { String value(); }功能:获取注解中定义的属性值
步骤:
获取注解定义的位置的对象 (例如:Class,Method,Field)
获取指定的注解(xxx.getAnnotation(注解.class),参数是该注解的Class对象
调用注解中的抽象方法获取配置的属性值
示例:通过注解执行Dog类的eatFood方法
//实体类 public class Dog { public void eatFood(){ System.out.println("eat...."); } } //自定义注解类 @Target(ElementType.TYPE) @Retention(RetentionPolicy.RUNTIME) public @interface Pro { //定义类和方法名 String className(); String methodName(); } //测试类 @Pro(className = "anno.an2.Dog",methodName = "eatFood") public class Test { public static void main(String[] args) throws Exception{ //1.获取该类的字节码文件 Class<Test> testClass = Test.class; //2.获取该类上的注解对象。通过Class对象 //如果想要获取方法上的注解,可以通过Method对象获取 Pro anno = testClass.getAnnotation(Pro.class); //3.调用注解对象上的抽象方法,获取返回值 String className = anno.className(); String methodName = anno.methodName(); System.out.println(className+"---"+methodName); //4.利用反射执行该类方法 Class<?> DaoClass = Class.forName(className); //4.1获取对象 Object dogObj = DaoClass.newInstance(); //4.2.获取方法 Method method = DaoClass.getMethod(methodName); //4.3执行方法 method.invoke(dogObj); } }