Java — 注解

Java 注解(Annotation)又称为 Java 标注,是 Java5 开始支持加入源代码的特殊语法元数据。

Java 语言中的类、方法、变量、参数和包等都可以被标注。

Java 标注可以通过反射获取标注的内容,在编译器生成 class 文件时,标注可以被嵌入到字节码中。

Java 虚拟机可以保留标注内容,在运行时可以获取到标注内容。

2.1、内置注解

Java 定义了一套注解,共有 10 个。5 个在 java.lang 包中,5 个在 java.lang.annotation 包中。

2.2、用在代码上的注解

注解 描述 @Override 检查该方法是否正确地重写了父类的方法。如果重写错误,会报编译错误。 @Deprecated 标记废弃的方法。如果使用该方法,会报编译警告。 @SuppressWarnings 提示编译器忽略注解中声明的警告。 @SafeVarargs Java 7 开始支持,忽略任何使用参数为泛型变量的方法或构造函数调用产生的警告。 @FunctionalInterface Java 8 开始支持,标识一个匿名函数或函数式接口。

2.3、用在其它注解上的注解

此类注解也称为 元注解(meta annotation)。

注解 描述 @Retention 标记在什么时候保存该注解信息,用于描述注解的生命周期。 @Target 标记该注解的使用范围。 @Documented 标记这些注解是否包含在用户文档中。 @Inherited 标记子类可以继承父类的注解。 @Repeatable Java 8 开始支持,标识某注解可以在同一个声明上使用多次。

格式: public @interface 注解名称 {}

步骤:创建注解、定义注解的参数和默认值、用上元注解配置该注解。

3.1、创建注解

示例:定义一个可用于检查字符串长度的注解

public @interface Length {

}

3.2、定义参数和默认值

public @interface Length {
    // 最小长度
    int min() default 0;
    // 最大长度
    int max() default Integer.MAX_VALUE;
    // 长度不合法
    String message() default "长度不合法";
}

注意:

3.3、用元注解配置注解

Retention 译为保留,该注解定义了一个注解的生命周期。

参数 描述 RetentionPolicy.RUNTIME (运行期间有效)注解可以保留到程序运行的时候,它会被加载进入到JVM中,所以在程序运行时可以获取到它们 RetentionPolicy.CLASS (编译期间有效)注解只被保留到编译进行的时候,它并不会被加载到JVM中 RetentionPolicy.SOURCE (源码期间有效)注解只在源码阶段保留,在编译器进行编译时它将被丢弃忽视

// 运行期间有效
@Retention(RetentionPolicy.RUNTIME)
public @interface Length {
    int min() default 0;
    int max() default Integer.MAX_VALUE;
    String message() default "长度不合法";
}

Target 译为目标,该是最为常用的元注解,可以指定自己能够被应用于源码中的哪些位置。

参数 描述 ElementType.TYPE 给一个类型进行注解,比如类、接口、枚举 ElementType.FIELD 给属性进行注解 ElementType.METHOD 给方法进行注解 ElementType.PARAMETER 给一个方法内的参数进行注解 ElementType.CONSTRUCTOR 给构造方法进行注解 ElementType.LOCAL_VARIABLE 给局部变量进行注解 ElementType.ANNOTATION_TYPE 给一个注解进行注解 ElementType.PACKAGE 给一个包进行注解

// 只用用于类属性或局部变量上
@Target({ElementType.FIELD, ElementType.LOCAL_VARIABLE})
public @interface Length {
    int min() default 0;
    int max() default Integer.MAX_VALUE;
    String message() default "长度不合法";
}

Documented 译为文档,标注了此注解的注解,能够将注解中的元素包含到 Javadoc 中去。

// 将该注解元素添加到JavaDoc中
@Documented
public @interface Length {
    int min() default 0;
    int max() default Integer.MAX_VALUE;
    String message() default "长度不合法";
}

Inherited 译为继承,使用该注解定义子类是否可继承父类定义的注解。

@Inherited 仅针对 @Target(ElementType.TYPE) 类型的注解有效,并且仅针对类的继承有效,对接口的继承无效。

// 使用该注解的类,其子类默认继承该注解(仅针对类的继承有效)
@Inherited
// 给一个类型进行注解,类、接口、枚举等
@Target(ElementType.TYPE)
public @interface Length {
    int min() default 0;
    int max() default Integer.MAX_VALUE;
    String message() default "长度不合法";
}

此时一个类用到了该注解

@Length(min = 2, max = 16, message = "昵称长度2~16之间")
public class Pet {}

其子类默认继承了该注解

// 相当于继承了@Length
public class Cat extends Pet {}

Repeatable 译为可重复,使用该元注解可以定义注解是否可重复。

// 作用于类的属性上
@Target(ElementType.FIELD)
// 注解Role注解,在类的属性上可使用多个Role注解
@Repeatable(Roles.class)
public @interface Role {
    String value() default "";
}
// 作用于类的属性上
@Target(ElementType.FIELD)
public @interface Roles {
    Role[] value();
}

@Repeatable 元注解标注了 @Role,而 @Repeatable 后面括号中的类相当于一个容器注解。

按照规定,它里面必须要有一个 value 的属性,属性类型是一个被 @Repeatable 注解标注过的注解数组。

经过 @Repeatable 修饰后,在某个类型声明处,就可以添加多个 @Role注解:

4.1、示例1

需求:使用 @length 注解判断学生姓名长度是否合法。

示例:

// 运行期间有效
@Retention(RetentionPolicy.RUNTIME)
// 作用于类的属性上
@Target({ElementType.FIELD})
public @interface Length {
    // 最小长度
    int min() default 0;
    // 最大长度
    int max() default Integer.MAX_VALUE;
    // 提示信息
    String message() default "长度不合法";
}
public class Student {

    @Length(min = 1, max = 3, message = "名称超长")
    private String name;

    public Student(String name) {
        this.name = name;
    }

    public String getName() {
        return name;
    }

    public static void main(String[] args) throws NoSuchFieldException {
        Student student = new Student("上官飞燕");
        // 判断name属性上length注解是否存在
        boolean lengthPresent = student.getClass().getDeclaredField("name").isAnnotationPresent(Length.class);
        if (lengthPresent) {
            // 获取注解内容
            Length length = student.getClass().getDeclaredField("name").getAnnotation(Length.class);
            // 使用注解信息校验名称是否合法
            if (length.min()

运行:

名称超长,合法长度:1~3。当前长度:4

4.2、示例2

需求:使用 @Repeatable 注解获取学生所有的职位。

示例:

// 运行期间有效
@Retention(RetentionPolicy.RUNTIME)
// 作用于类的属性上
@Target(ElementType.FIELD)
public @interface Roles {
    Role[] value();
}
// 运行期有效
@Retention(RetentionPolicy.RUNTIME)
// 作用与类的属性上
@Target({ElementType.FIELD})
// 注解Role注解,在类的属性上可使用多Role注解
@Repeatable(Roles.class)
public @interface Role {
    String value() default "";
}
public class Student {

    @Role("学习委员")
    @Role("体育委员")
    private String name;

    public Student(String name) {
        this.name = name;
    }

    public String getName() {
        return name;
    }

    public static void main(String[] args) throws NoSuchFieldException {
        Student student = new Student("王五");
        // 判断name属性上@Role注解是否存在
        Role[] roles = student.getClass().getDeclaredField("name").getAnnotationsByType(Role.class);
        if (roles.length == 0) {
            System.out.println("Student.name属性上没有找到@Role注解");
        } else {
            // 获取注解内容
            for (Role role : roles) {
                System.out.println(student.getName() + " = " + role.value());
            }
        }
    }
}

运行:

王五 = 学习委员
王五 = 体育委员

Original: https://www.cnblogs.com/bybeiya/p/15086690.html
Author: 北涯
Title: Java — 注解

原创文章受到原创版权保护。转载请注明出处:https://www.johngo689.com/585919/

转载文章受原作者版权保护。转载请注明原作者出处!

(0)

大家都在看

  • .NET服务治理之限流中间件-FireflySoft.RateLimit

    FireflySoft.RateLimit自2021年1月发布第一个版本以来,经历了多次升级迭代,目前已经十分稳定,被很多开发者应用到了生产系统中,最新发布的版本是3.0.0。 它…

    Linux 2023年6月13日
    0101
  • shell大全定时清空文件内容,定时记录文件大小

    shell大全定时清空文件内容,定时记录文件大小 bash;gutter:true;</p> <h1>!/bin/bash</h1> <h…

    Linux 2023年5月28日
    0128
  • Mybatis源码解读-SpringBoot中配置加载和Mapper的生成

    本文 mybatis-spring-boot探讨在springboot工程中mybatis相关对象的注册与加载。 建议先了解mybatis在spring中的使用和springboo…

    Linux 2023年6月7日
    0114
  • Redisson和spring-integration-redis 分布式锁

    Redisson 引入依赖 配置文件application.properties 配置类 配置文件application.properties(这种方式完全兼容SpringBoot…

    Linux 2023年5月28日
    088
  • typora使用

    一:typora、配置picgo+github安装以及配置 picgo 、Node.js和typora安装包见百度网盘链接: 参考的教程如下: 二:typora使用语法 Ctrl+…

    Linux 2023年6月13日
    090
  • CA证书介绍与格式转换

    PKCS 公钥加密标准(Public Key Cryptography Standards, PKCS),此一标准的设计与发布皆由RSA资讯安全公司(英语:RSA Security…

    Linux 2023年6月6日
    088
  • Jmeter环境变量配置你不得不知道的事情

    在安装Jmeter的过程中大家肯定需要配置环境,但是为什么要配置JDK的环境变量呢?大家有没有好奇过,有没有仔细去像一下呢,其实在安装Jmeter前,大家应该都知道Jmeter是我…

    Linux 2023年6月14日
    0113
  • 以Docker方式安装Redis集群

    以 Redis-6.0.6 为例,先从仓库将镜像拉下来: docker pull redis:6.0.6 Redis 的配置文件和数据文件不能放在镜像中,这里选择容器中的目录和宿主…

    Linux 2023年5月28日
    071
  • 挂载mount

    挂载mount 1.查看系统挂载的磁盘情况 df df -h 2.挂载:mount 将光驱挂在到/mnt目录: mount /dev/cdrom /mnt #mount 准备挂载的…

    Linux 2023年6月11日
    095
  • 一、Linux系统的简介与历史发展

    1.操作系统的介绍内核负责控制硬件资源分配,而如果只有内核,则只能让计算机硬件运行,而不能有任何功能,因此需要系统调用提供给开发者使用,从而开发应用程序;内核能够控制硬件,比如:让…

    Linux 2023年6月7日
    0153
  • Redis具体解释

    2、 redis.windows.conf各项配置參数介绍 默认情况下,redis不是在后台模式执行的,假设须要在后台进程执行。把该项的值更改为yes。默觉得no daemoniz…

    Linux 2023年5月28日
    0100
  • 【原创】Linux虚拟化KVM-Qemu分析(九)之virtio设备

    背景 Read the fucking source code! –By 鲁迅 A picture is worth a thousand words. –…

    Linux 2023年6月8日
    073
  • 快速掌握 Base 64 | Java JS 密码系列

    Java 密码系列 – Java 和 JS Base 64 Base 64 不属于密码技术,仅是编码方式。但由于在 Java、JavaScript、区块链等出现的频率较…

    Linux 2023年6月7日
    097
  • Linux系统优化

    一、 系统信息查看方法 查看系统名称信息 cat /etc/redhat-release CentOS Linux release 7.9.2009 (Core) 查看系统内核版本…

    Linux 2023年5月27日
    0137
  • Linux Centos7.5 vsftp 的安装与配置

    安装及配置 安装 sudo yum install vsftpd -y 服务管理 启动服务 service vsftpd start 关闭服务 service vsftpd sto…

    Linux 2023年6月14日
    083
  • CAPL学习笔记

    CAPL是CANOE自带的一种编程语言,要和CANOE中的一个节点绑定在一起。它的文件后缀是.can。 两种添加方式:1. 在simulation setup中增加一个网络节点,配…

    Linux 2023年6月13日
    085
亲爱的 Coder【最近整理,可免费获取】👉 最新必读书单  | 👏 面试题下载  | 🌎 免费的AI知识星球