1.什么是注解?

作者: 还算年轻 | 来源:发表于2020-08-05 17:41 被阅读0次

一、理解Java注解

    实际上Java注解与普通修饰符(public、static、void等)的使用方式并没有多大区别,起到标注的作用。

    下面的例子是常见的注解:

public class AnnotationDemo {

    //@Test注解修饰方法A

    @Test

    public static void A(){System.out.println("Test.....");}

    //一个方法上可以拥有多个不同的注解

    @Deprecated

    @SuppressWarnings("uncheck")

    public static void B(){}

}

通过在方法上使用@Test注解后,在运行该方法时,测试框架会自动识别该方法并单独调用,@Test实际上是一种标记注解,起标记作用,运行时告诉测试框架该方法为测试方法。

而对于@Deprecated和@SuppressWarnings(“uncheck”),则是Java本身内置的注解。

二、基本语法

声明注解与元注解

我们先来看看前面的Test注解是如何声明的:

//声明Test注解

@Target(ElementType.METHOD)  

@Retention(RetentionPolicy.RUNTIME)

public @interface Test {

}

@interface声明了Test注解,并使用@Target注解传入ElementType.METHOD参数来标明@Test只能用于方法上,@Retention(RetentionPolicy.RUNTIME)则用来表示该注解生存期是运行时,从代码上看注解的定义很像接口的定义,确实如此,毕竟在编译后也会生成Test.class文件。对于@Target和@Retention是由Java提供的元注解,所谓元注解就是标记其他注解的注解

元注解:@Target 用来约束注解可以应用的地方(如方法、类或字段)

     /**标明该注解可以用于类、接口(包括注解类型)或enum声明*/

    TYPE,

    /** 标明该注解可以用于字段(域)声明,包括enum实例 */

    FIELD,

    /** 标明该注解可以用于方法声明 */

    METHOD,

    /** 标明该注解可以用于参数声明 */

    PARAMETER,

    /** 标明注解可以用于构造函数声明 */

    CONSTRUCTOR,

    /** 标明注解可以用于局部变量声明 */

    LOCAL_VARIABLE,

    /** 标明注解可以用于注解声明(应用于另一个注解上)*/

    ANNOTATION_TYPE,

    /** 标明注解可以用于包声明 */

    PACKAGE,

    /*** 标明注解可以用于类型参数声明(1.8新加入) * @since 1.8 */

    TYPE_PARAMETER, java8

    /**类型使用声明(1.8新加入)@since 1.8*/

    TYPE_USE java8

当注解未指定Target值时,则此注解可以用于任何元素之上,多个值使用{}包含并用逗号隔开,如下:

@Target(value={CONSTRUCTOR, FIELD, LOCAL_VARIABLE, METHOD, PACKAGE, PARAMETER, TYPE})

元注解:@Retention用来约束注解的生命周期,分别有三个值,源码级别(source),类文件级别(class)或者运行时级别(runtime),其含有如下:

SOURCE:注解将被编译器丢弃(该类型的注解信息只会保留在源码里,源码经过编译后,注解信息会被丢弃,不会保留在编译好的class文件里)

CLASS:注解在class文件中可用,但会被VM丢弃(该类型的注解信息会保留在源码里和class文件里,在执行的时候,不会加载到虚拟机中),请注意,当注解未定义Retention值时,默认值是CLASS,如Java内置注解,@Override、@Deprecated、@SuppressWarnning等

RUNTIME:注解信息将在运行期(JVM)也保留,因此可以通过反射机制读取注解的信息(源码、class文件和执行的时候都有注解的信息),如SpringMvc中的@Controller、@Autowired、@RequestMapping等。

元注解:@Documented 被修饰的注解会生成到javadoc中。

元注解:@Inherited 可以让注解被继承,但这并不是真的继承,只是通过使用@Inherited,可以让子类Class对象使用getAnnotations()获取父类被@Inherited修饰的注解。


三、注解元素及其数据类型

通过上述对@Test注解的定义,我们了解了注解定义的过程,由于@Test内部没有定义其他元素,所以@Test也称为标记注解(marker annotation),但在自定义注解中,一般都会包含一些元素以表示某些值,方便处理器使用,这点在下面的例子将会看到:

@Target(ElementType.TYPE)//只能应用于类上

@Retention(RetentionPolicy.RUNTIME)//保存到运行时

public @interface DBTable {

    String name() default "";

}

上述定义一个名为DBTable的注解,该用于主要用于数据库表与Bean类的映射(稍后会有完整案例分析),与前面Test注解不同的是,我们声明一个String类型的name元素,其默认值为空字符,但是必须注意到对应任何元素的声明应采用方法的声明方式,同时可选择使用default提供默认值,@DBTable使用方式如下:

//在类上使用该注解

@DBTable(name = "MEMBER")

public class Member {

    //.......

}

关于注解支持的元素数据类型除了上述的String,还支持如下数据类型

所有基本类型(int,float,boolean,byte,double,char,long,short)

String

Class

enum

Annotation

上述类型的数组

倘若使用了其他数据类型,编译器将会丢出一个编译错误,注意,声明注解元素时可以使用基本类型但不允许使用任何包装类型,同时还应该注意到注解也可以作为元素的类型,也就是嵌套注解。

@Target(ElementType.TYPE)

@Retention(RetentionPolicy.RUNTIME)

@interface Reference{

    boolean next() default false;

}

public @interface AnnotationElementDemo {

    //枚举类型

    enum Status {FIXED,NORMAL};

    //声明枚举

    Status status() default Status.FIXED;

    //布尔类型

    boolean showSupport() default false;

    //String类型

    String name()default "";

    //class类型

    Class<?> testCase() default Void.class;

    //注解嵌套

    Reference reference() default @Reference(next=true);

    //数组类型

    long[] value();

}

编译器对默认值的限制

编译器对元素的默认值有些过分挑剔。首先,元素不能有不确定的值。也就是说,元素必须要么具有默认值,要么在使用注解时提供元素的值。其次,对于非基本类型的元素,无论是在源代码中声明,还是在注解接口中定义默认值,都不能以null作为值,这就是限制,没有什么利用可言,但造成一个元素的存在或缺失状态,因为每个注解的声明中,所有的元素都存在,并且都具有相应的值,为了绕开这个限制,只能定义一些特殊的值,例如空字符串或负数,表示某个元素不存在。

注解不支持继承

注解是不支持继承的,因此不能使用关键字extends来继承某个@interface,但注解在编译后,编译器会自动继承java.lang.annotation.Annotation接口,这里我们反编译前面定义的DBTable注解

package com.zejian.annotationdemo;

import java.lang.annotation.Annotation;

//反编译后的代码

public interface DBTable extends Annotation

{

    public abstract String name();

}

快捷方式

所谓的快捷方式就是注解中定义了名为value的元素,并且在使用该注解时,如果该元素是唯一需要赋值的一个元素,那么此时无需使用key=value的语法,而只需在括号内给出value元素所需的值即可。这可以应用于任何合法类型的元素,记住,这限制了元素名必须为value.

//定义注解

@Target(ElementType.FIELD)

@Retention(RetentionPolicy.RUNTIME)

@interface IntegerVaule{

    int value() default 0;

    String name() default "";

}

//使用注解

public class QuicklyWay {

    //当只想给value赋值时,可以使用以下快捷方式

    @IntegerVaule(20)

    public int age;

    //当name也需要赋值时必须采用key=value的方式赋值

    @IntegerVaule(value = 10000,name = "MONEY")

    public int money;

}

相关文章

  • 1.什么是注解?

    一、理解Java注解 实际上Java注解与普通修饰符(public、static、void等)的使用方式并没有多大...

  • Annotation&Junit

    注解测试XML 1. 什么是注解 所有的Annotation都是java.lang.annotation.Anno...

  • #注解、反射及动态代理

    一、注解(Annotation) 1.什么是注解:   注解可以说是注释的更高级的一种,相当于标记,注解同样不影响...

  • Android进阶 注解

    Android进阶 注解 1.注解是什么 注解是一系列元数据,它提供数据用来解释程序代码,但是注解并非是所解释的代...

  • lombok 注解

    lombok 注解 1. 什么是 lombok 注解 Lombok 是一种 Java™ 实用工具,可用来帮助开发人...

  • java面试题(自制版)

    注解(什么是注解,注解的作用,注解如何使用、注解的原理) 什么是注解? 注解是一种代码的表现形式,通过在inter...

  • Java注解

    1.元注解 元注解是指注解的注解。包括 @Retention @Target @Document @Inheri...

  • Java中的注解

    1. 注解 注解是JDK5.0才开始引入的技术. 1.1 什么是注解 以下引用维基百科的解释,看不懂也没事,我会做...

  • Java-深入了解元注解

    1. 简介 元注解是指注解的注解,包括@Retention @Target @Document @Inherite...

  • 中级15 - Java的注解

    注解使程序更加简洁,花更少的力气完成更多的工作——这也是计算机的意义所在。 什么是注解 运行时获取注解信息 1. ...

网友评论

    本文标题:1.什么是注解?

    本文链接:https://www.haomeiwen.com/subject/hororktx.html