注解详解

Lou.Chen
大约 5 分钟

一、什么是注解(Annotation)

​ Java注解又称Java标注,是Java语言5.0版本开始支持加入源代码的特殊语法元数据。为我们在代码中添加信息提供了一种形式化的方法,使我们可以在稍后某个时刻非常方便的使用这些数据。 ​ Java语言中的类、方法、变量、参数和包等都可以被标注。和Javadoc不同,Java标注可以通过反射获取注解内容。在编译器生成类文件时,注解可以被嵌入到字节码中。Java虚拟机可以保留注解内容,在运行时可以获取到注解内容。

二、内置注解

java 定义了一套注解,共有 7 个,3 个在 java.lang 中,剩下 4 个在 java.lang.annotation 中。

  • @Override: 限定重写父类方法, 该注解只能用于方法
  • @Deprecated: 用于表示所修饰的元素(类, 方法等)已过时。通常是因为 所修饰的结构危险或存在更好的选择
  • @SuppressWarnings: 抑制编译器警告

1、作用在代码的注解是

  • @Override - 检查该方法是否是重写方法。如果发现其父类,或者是引用的接口中并没有该方法时,会报编译错误。
  • @Deprecated - 标记过时方法。如果使用该方法,会报编译警告。
  • @SuppressWarnings - 指示编译器去忽略注解中声明的警告。

2、作用在其他注解的注解(或者说元注解)是:

  • @Retention - 标识这个注解怎么保存,是只在代码中,还是编入class文件中,或者是在运行时可以通过反射访问。
  • @Documented - 标记这些注解是否包含在用户文档中。
  • @Target - 标记这个注解应该是哪种 Java 成员。
  • @Inherited - 标记这个注解是继承于哪个注解类(默认 注解并没有继承于任何子类)

3、从 Java 7 开始,额外添加了 3 个注解:

  • @SafeVarargs - Java 7 开始支持,忽略任何使用参数为泛型变量的方法或构造函数调用产生的警告。
  • @FunctionalInterface - Java 8 开始支持,标识一个匿名函数或函数式接口。
  • @Repeatable - Java 8 开始支持,标识某注解可以在同一个声明上使用多次。

三、元注解

1、@Retention

@Retention annotation指定标记注释的存储方式:

  • RetentionPolicy.SOURCE - 标记的注释仅保留在源级别中,并由编译器忽略。
  • RetentionPolicy.CLASS - 标记的注释在编译时由编译器保留,即会生成class文件,但是不会加载到内存中。默认行为。
  • RetentionPolicy.RUNTIME - 标记的注释由JVM保留**,会生成class文件,也会加载到内存中,可由反射获取。**

2、@Documented

@Documented 注释表明,无论何时使用指定的注释,都应使用Javadoc工具记录这些元素。(默认情况下,注释不包含在Javadoc中。)有关更多信息,请参阅 Javadoc工具页面。

3、@Target

@Target 注释标记另一个注释,以限制可以应用注释的Java元素类型。目标注释指定以下元素类型之一作为其值

  • ElementType.TYPE 可以应用于类的任何元素。
  • ElementType.FIELD 可以应用于字段或属性。
  • ElementType.METHOD 可以应用于方法级注释。
  • ElementType.PARAMETER 可以应用于方法的参数。
  • ElementType.CONSTRUCTOR 可以应用于构造函数。
  • ElementType.LOCAL_VARIABLE 可以应用于局部变量。
  • ElementType.ANNOTATION_TYPE 可以应用于注释类型。
  • ElementType.PACKAGE 可以应用于包声明。
  • ElementType.TYPE_PARAMETER 表示该注解能写在类型变量的声明语 句中(如:泛型声明)
  • ElementType.TYPE_USE 表示该注解能写在使用类型的任何语句中。如:变量修饰,异常修饰

4、@Inherited

@Inherited 注释表明注释类型可以从超类继承。当用户查询注释类型并且该类没有此类型的注释时,将查询类的超类以获取注释类型(默认情况下不是这样)。此注解仅适用于类声明。

5、@Repeatable

Repeatable Java SE 8中引入的,@Repeatable注释表明标记的注释可以多次应用于相同的声明或类型使用(即可以重复在同一个类、方法、属性等上使用)。

@Inherited
@Documented
//用于类和方法上
@Target({ElementType.TYPE,ElementType.METHOD})
//运行时可获取
@Retention(RetentionPolicy.RUNTIME)
public @interface MyAnnotations {
    MyAnnotaion[] value();
}

@Inherited
@Documented
//用于类和方法上
@Target({ElementType.TYPE, ElementType.METHOD})
//运行时可获取
@Retention(RetentionPolicy.RUNTIME)
//声明当前注解为可重复注解
@Repeatable(value = MyAnnotations.class)
public @interface MyAnnotaion {
    //属性名value  属性类型String  默认值 hi
    String value() default "hi";
}

//jdk1.8之前使用多个相同注解的方式
//@MyAnnotations({@MyAnnotaion(value = "haha"),@MyAnnotaion(value = "xixi")})
//jdk.18之后使用重复注解的方式
@MyAnnotaion("haha")
@MyAnnotaion("xixi")
public class Pserson {
}

四、基本使用

https://www.jianshu.com/p/a7bedc771204open in new window

1、使用于属性上

@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface MyField {
    String descirption();
    int length() default 2;
}
public class Person {
    @MyField(descirption = "姓名",length = 20)
    private String name;
    @MyField(descirption = "年龄")
    private int age;
}
public class Main {
    public static void main(String[] args) {
        //得到应用于注解上的Class类
        Class personClass = Person.class;
        //获取运行时类的所有字段
        Field[] declaredFields = personClass.getDeclaredFields();
        for (Field field : declaredFields) {
            //判断该属性上是否有指定的注解
            if (field.isAnnotationPresent(MyField.class)) {
                //获取当前属性的注解对象
                MyField annotation = field.getAnnotation(MyField.class);
                //获得注解信息
                //name,姓名,20
                //age,年龄,2
                System.out.println(field.getName()+","+annotation.descirption()+","+annotation.length());
            }
        }
    }
}

2、使用于类上

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
public @interface MyClass {
    String description();
}
@MyClass(description = "人")
public class Person {
}
public class Main {
    public static void main(String[] args) {
        Class personClass = Person.class;
        //判断当前类是否有MyClass注解
        if (personClass.isAnnotationPresent(MyClass.class)) {
            MyClass annotation = (MyClass) personClass.getAnnotation(MyClass.class);
            //人
            System.out.println(annotation.description());
        }
    }
}

3、使用于方法上

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface MyMethod {
    String description();
}
public class Person{
    @MyMethod(description = "打招呼方法...")
    public void say() {
        System.out.println("hello...");
    }

    @MyMethod(description = "获取密码方法...")
    public String getPassword() {
        return "123456";
    }
}
public class Main {
    public static void main(String[] args) throws IllegalAccessException, InstantiationException, InvocationTargetException {
        Class personClass = Person.class;
        Person person = ((Person) personClass.newInstance());
        Method[] declaredMethods = personClass.getMethods();
        for (Method method : declaredMethods) {
            if (method.isAnnotationPresent(MyMethod.class)) {
                MyMethod annotation = method.getAnnotation(MyMethod.class);
                System.out.println(method.invoke(person)+"---"+annotation.description());
            }
        }
    }
}
hello...
null---打招呼方法...
123456---获取密码方法...