Java——注解详解

注解

Java 注解(Annotation)又称 Java 标注,是 JDK5.0 引入的一种注释机制。

Java 语言中的类、方法、变量、参数和包等都可以被标注。和注释不同,Java 标注可以通过反射获取标注内容。在编译器生成类文件时,标注可以被嵌入到字节码中。Java 虚拟机可以保留标注内容,在运行 时可以获取到标注内容 。 当然它也支持自定义 Java 标注。

  • 主要用于:
    • 编译格式检查
    • 反射中解析
    • 生成帮助文档
    • 跟踪代码依赖

内置注解

  • @Override:重写 *

    • 定义在java.lang.Override
  • @Deprecated:废弃 *

    • 定义在java.lang.Deprecated
  • @SafeVarargs:

    • Java 7开始支持,忽略任何使用参数为泛型变量的方法或构造函数调用产生的警告。
  • @Functionallnterface:函数式接口 *

    • 从Java 8开始支持,标识一个匿名函数或函数式接口。
  • @Repeatable:标识某注解可以在同一个声明上使用多次

    • 从java 8开始支持
  • SuppressWarnings:抑制编译时的警告信息 *

    • 定义在java.lang.SuppressWarnings
    • 三种使用方式
    1.@SuppressWarnings("unchecked") [^ 抑制单类型的警告]
    2.@SuppressWarnings("unchecked","rawtypes") [^ 抑制多类型的警告]
    3.@SuppressWarnings("all") [^ 抑制所有类型的警告]
    
关键字
all抑制所有警告
boxing抑制装箱、拆箱操作时候的警告
cast抑制映射相关的警告
dep-ann抑制启用注释的警告
deprecation抑制过期方法警告
fallthrough抑制确在switch中缺失breaks的警告
finally抑制finally模块没有返回的警告
hiding抑制相对于隐藏变量的局部变量的警告
incomplete-switch忽略没有完整的switch语句
nls忽略非nls格式的字符
null忽略对null的操作
rawtypes使用generics时忽略没有指定相应的类型
restriction抑制禁止使用劝阻或禁止引用的警告
serial忽略在serializable类中没有声明serialVersionUID变量
static-access抑制不正确的静态访问方式警告
synthetic-access抑制子类没有按最优方法访问内部类的警告
unchecked抑制没有进行类型检查操作的警告
unqualified-field-access抑制没有权限访问的域的警告
unused抑制没被使用过的代码的警告

元注解

作用在其他注解的注解

  • @Retention - 标识这个注解怎么保存,是只在代码中,还是编入class文件中,或者是在运行时可 以通过反射访问。

  • @Documented - 标记这些注解是否包含在用户文档中 javadoc。

  • @Target - 标记这个注解应该是哪种 Java 成员。

  • @Inherited - 标记这个注解是自动继承的

    • 1. 子类会继承父类使用的注解中被@Inherited修饰的注解
      2. 接口继承关系中,子接口不会继承父接口中的任何注解,不管父接口中使用的注解有没有
      被@Inherited修饰
      3. 类实现接口时不会继承任何接口中定义的注解
      

自定义注解

架构

在这里插入图片描述

  1. Annotation与RetentionPolicy 与ElementType 。
  • 每 1 个 Annotation 对象,都会有唯一的 RetentionPolicy 属性;至于 ElementType 属性,则有 1~n 个。
  1. ElementType(注解的用途类型)

    “每 1 个 Annotation” 都与 “1~n 个 ElementType” 关联。当 Annotation 与某个 ElementType 关联 时,就意味着:Annotation有了某种用途。例如,若一个 Annotation 对象是 METHOD 类型,则该 Annotation 只能用来修饰方法。

    public enum ElementType {
        /* 类、接口(包括注释类型)或枚举声明 */
        TYPE,
    
        /* 字段声明(包括枚举常量) */
        FIELD,
    
        /* 方法声明 */
        METHOD,
    
        /* 参数声明 */
        PARAMETER,
    
        /* 构造方法声明 */
        CONSTRUCTOR,
    
        /* 局部变量声明 */
        LOCAL_VARIABLE,
    
        /* 注释类型声明 */
        ANNOTATION_TYPE,
    
        /* 包声明 */
        PACKAGE,
    
        /**
         * Type parameter declaration
         *
         * @since 1.8
         */
        TYPE_PARAMETER,
    
        /**
         * Use of a type
         *
         * @since 1.8
         */
        TYPE_USE,
    
        /**
         * Module declaration.
         *
         * @since 9
         */
        MODULE
    }
    
  2. RetentionPolicy(注解作用域策略)。

​ 每 1 个 Annotation" 都与 “1 个 RetentionPolicy” 关联

  • 若 Annotation 的类型为 SOURCE,则意味着:Annotation 仅存在于编译器处理期间,编译器处理完之后,该Annotation 就没用了。例如," @Override" 标志就是一个 Annotation。当它修饰一个方法的时候,就意味着该方法覆盖父类的方法;并且在编译期间会进行语法检查!编译器处 理完后,"@Override" 就没有任何作用了。
  • 若 Annotation的类型为 CLASS,则意味着:编译器将Annotation存储于类对应的 .class 文件中,它是Annotation的默认行为。
  • 若 Annotation 的类型为 RUNTIME,则意味着:编译器将 Annotation 存储于 class 文件中,并 且可由JVM读入。
public enum RetentionPolicy {
    /* Annotation信息仅存在于编译器处理期间,编译器处理完之后就没有该
Annotation信息了 */
    SOURCE,

    /* 编译器将Annotation存储于类对应的.class文件中。默认行为 */
    CLASS,

    /* 编译器将Annotation存储于class文件中,并且可由JVM读入 */
    RUNTIME
}
定义格式
  • @interface 自定义注解名{}
注意事项
  1. 定义的注解,自动继承了java.lang,annotation.Annotation接口
  2. 注解中的每一个方法,实际是声明的注解配置参数
    • 方法的名称就是 配置参数的名称
    • 方法的返回值类型,就是配置参数的类型。只能是:基本类型/Class/String/enum
  3. 可以通过default来声明参数的默认值
  4. 如果只有一个参数成员,一般参数名为value
  5. 注解元素必须要有值,我们定义注解元素时,经常使用空字符串、0作为默认值。
案例
//注解是否包含在文档中
@Documented
//用途类型
@Target({ElementType.TYPE,ElementType.METHOD})
//保存策略
@Retention(RetentionPolicy.RUNTIME)
//可以继承
@Inherited
@interface MyAnnotation{
    String[] value() default  "张三";
    int num() default 1;
}

上面的作用是定义一个 Annotation,我们可以在代码中通过 “@MyAnnotation1” 来使用它。

@Documented, @Target, @Retention, @interface 都是来修饰 MyAnnotation1 的。

  1. @interface

    使用 @interface 定义注解时,意味着它实现了java.lang.annotation.Annotation 接口,即该注解就是 一个Annotation。

    定义Annotation时,@interface是必须的。

    注意:它和我们通常的implemented实现接口的方法不同。Annotation 接口的实现细节都由编译器完成。通过@interface 定义注解后,该注解不能继承其他的注解或接口。

  2. @Documented

类和方法的 Annotation在缺省情况下是不出现在 javadoc 中的。如果使用@Documented 修饰该 Annotation,则表示它可以出现在 javadoc 中。

定义Annotation时,@Documented可有可无;若没有定义,则 Annotation不会出现在javadoc中。

  1. @Target(ElementType.TYPE)

    前面我们说过,ElementType是Annotation的类型属性。而@Target的作用,就是来指定Annotation的类型属性。

    @Target(ElementType.TYPE) 的意思就是指定该 Annotation 的类型是 ElementType.TYPE。这就意味着,MyAnnotation1是来修饰"类、接口(包括注释类型)或枚举声明"的注解。

    定义 Annotation 时,@Target可有可无。若有@Target,则该 Annotation 只能用于它所指定的地方;若没有 @Target,则该 Annotation 可以用于任何地方。

  2. @Retention(RetentionPolicy.RUNTIME)

    前面我们说过,RetentionPolicy是 Annotation 的策略属性,而 @Retention的作用,就是指定Annotation的策略属性。

    @Retention(RetentionPolicy.RUNTIME) 的意思就是指定该 Annotation 的策略是 RetentionPolicy.RUNTIME。这就意味着,编译器会将该Annotation信息保留在 .class文件中,并且能被虚拟机读取。

    定义Annotation时,@Retention 可有可无。若没有@Retention,则默认是 RetentionPolicy.CLASS

notation的策略属性。

@Retention(RetentionPolicy.RUNTIME) 的意思就是指定该 Annotation 的策略是 RetentionPolicy.RUNTIME。这就意味着,编译器会将该Annotation信息保留在 .class文件中,并且能被虚拟机读取。

定义Annotation时,@Retention 可有可无。若没有@Retention,则默认是 RetentionPolicy.CLASS

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值