注解【开发实践】

文章目录

    • 一、注解概述
      • 1.1 什么是注解
      • 1.2 注解的作用
      • 1.3 一些特殊的注解
    • 二、元注解
      • 2.1 @Retention
      • 2.2 @target
      • 2.3 @Documented
      • 2.4 @Inherited
      • 2.5 @Repeatable
    • 三、注解的使用
      • 3.1 定义注解
      • 3.2 编写注解处理器
      • 3.3 注册注解处理器

一、注解概述

1.1 什么是注解

注解(Annotation)是编程语言中的一种元数据形式,它提供了一种机制,可以在不直接影响程序代码执行逻辑的情况下,向源代码中添加信息。注解不会直接改变程序的行为,而是作为一种标记或指令,供编译器、开发工具或运行时环境解析和使用,以实现特定的功能或处理。

简单来说,使用注解标注相当于在代码的某个位置打上标签,标签上有一些元数据,仅此而已。注解本身不改变程序行为,需要通过自定义的注解处理器来扫描注解,并在扫描到注解后进行预设的处理。

1.2 注解的作用

  • 生成文档:注解可以用来生成API文档,如Java中的@param、@return等,帮助自动生成详细的文档说明。
  • 编译时检查:例如,@Override注解告诉编译器该方法意在重写超类的方法,如果实际上没有匹配的方法,则编译器会报错。
  • 编译时处理:注解处理器可以在编译阶段读取注解信息,并据此生成额外的源代码或资源文件,如Lombok的@Data自动为类生成getter和setter方法。
  • 运行时处理:某些注解在程序运行时会被解析,用于动态地决定程序行为,如Spring框架中使用注解来配置依赖注入、事务管理等。
  • 代码分析和框架集成:注解可以帮助工具和框架识别代码结构,如单元测试框架识别测试方法,或IDE识别需要特殊处理的代码块。

1.3 一些特殊的注解

标记注解:没有元素的注解。
单值注解:只有一个元素的注解,设置元素值时可以省略元素名。
元注解:作用于注解的注解。
容器注解:为定义可重复使用注解而定义的辅助注解。

二、元注解

元注解是Java中一种特殊的注解,它们主要用于注解其他的注解,为这些注解提供元数据级别的信息,如生命周期、目标位置、是否可继承等。Java自身定义了几个标准的元注解,它们位于java.lang.annotation包中。

2.1 @Retention

指定被修饰的注解的保留策略,即指定注解的生命周期。它有一个枚举类型的参数RetentionPolicy,可选值有:

  • SOURCE:注解只保留在源码中,编译时会被忽略。
  • CLASS(默认):注解会保留在class文件中,但运行时不会被虚拟机保留,因此无法在运行时通过反射访问。
  • RUNTIME:注解会保留在class文件中,并且能通过反射在运行时访问。

2.2 @target

限定被修饰的注解能够应用到哪些程序元素上。它的参数是一个枚举类型ElementType,可选值包括TYPE(类,接口,枚举)、METHOD(方法)、FIELD(字段)、PARAMETER(参数)、CONSTRUCTOR(构造器)、LOCAL_VARIABLE(局部变量)、ANNOTATION_TYPE(注解)、PACKAGE(包)等。

如果想将注解用于与多种程序元素上,可以使用多个值,这些值之间使用逗号隔开。默认情况下(没有指定时)可以应用于上述所有程序元素。

2.3 @Documented

标记被修饰的注解,使其在生成JavaDoc文档时包含进去。

简单来说,如果一个注解被@Documented修饰,那么这个注解所标注的元素的API文档中将展示这个注解,否则不会展示。

2.4 @Inherited

标记一个注解,表示该注解支持子类继承(仅对类继承有效,不适用于接口继承)。当一个类上加了该注解后,该注解也会隐式地应用于这个类的所有子类。

2.5 @Repeatable

表示某个注解可以重复使用在同一个目标上,此种注解的定义需要配合容器注解来使用,容器注解用于存储可重复注解的实例数组(详情见后)。

三、注解的使用

3.1 定义注解

定义注解就像定义接口,但有所区别。关键字换成了@interface,实现体中类似定义"方法"的形式来定义元素:"返回值"类型即元素类型,"方法名"即元素名,注意没有参数和方法体。对于每个元素,可以使用default来设置默认值。

定义不可重复使用的注解。

public @interface MyAnnotation {

    // 没有默认值的情况,使用时必须提供
    String requiredValue();
    
    // 有默认值的情况,使用时可以不提供
    String optionalValue() default "Default Value";
}

定义可重复使用的注解

// 定义可重复使用的注解
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
// 使用@Repeatable标记,表明是可重复使用的注解,使用属性指对应的容器注解。
@Repeatable(MyAnnotations.class)
public @interface MyAnnotation {
    String value();
}

// 定义容器注解,就是一种特殊的注解,用于存储可重复使用注解的实例数组。
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
public @interface MyAnnotations {
	// 可重复使用注解的数组类型
    MyAnnotation[] myAnnotations();
}

3.2 编写注解处理器

编写注解处理器,需要实现javax.annotation.processing.Processor接口。最简单的实现可以继承AbstractProcessor 抽象类来实现,其中process()方法是必须实现的,负责实际的注解处理逻辑。

process的参数
annotations:注解处理器所负责的注解的集合。
roundEnvironment:提供了访问注解所应用元素的环境。

process的返回值
返回true表示该处理器已经处理了所负责的所有注解,不需要其他处理器再处理这些注解。返回false则表示后续处理器还可以尝试处理这些注解。在大多数情况下,如果处理器完成了它的工作,应该返回true。

// 指定注解处理器需要处理的注解
@SupportedAnnotationTypes("你的包名.自定义注解名")
// 指定注解处理器支持的Java语言版本。
@SupportedSourceVersion(SourceVersion.RELEASE_8)
public class MyAnnotationProcessor extends AbstractProcessor {

    @Override
    public synchronized void init(ProcessingEnvironment processingEnv) {
        super.init(processingEnv);
        // 初始化处理环境,可在此处设置工具、日志等
    }

     @Override
    public boolean process(Set<? extends TypeElement> annotations, RoundEnvironment roundEnv) {
    	// 遍历需要处理的注解
        for (TypeElement annotation : annotations) {
        	// 对于每个注解,获取其标注的所有对象
            Set<? extends Element> annotatedElements = roundEnv.getElementsAnnotatedWith(annotation);
            // 遍历注解标注的对象,进行相应的处理
            for (Element element : annotatedElements) {
                ...
            }
        }
        return true;
    }
}

3.3 注册注解处理器

确保你的处理器能够被Java编译器发现,通常这需要在项目的META-INF/services/javax.annotation.processing.Processor文件中声明处理器的全限定名。如果你使用的是Maven或Gradle等构建工具,通常有相应的配置方式来自动注册处理器。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/772112.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

汇聚荣拼多多评价好不好?

汇聚荣拼多多评价好不好?在探讨电商平台的口碑时&#xff0c;用户评价是衡量其服务质量和商品质量的重要指标。拼多多作为国内领先的电商平台之一&#xff0c;其用户评价自然成为消费者选择购物平台时的参考依据。针对“汇聚荣拼多多评价好不好?”这一问题&#xff0c;可以从…

Elasticsearch初识与 index+mapping+document 基操

前言 在21年多少有使用过es 当时是在艺术赛道的一个教育公司&#xff0c;大概流程就是 将mysql中的各种课程数据通过logstash汇总到es 然后提供rest接口出去。由于在职时间较短(很不幸赶上了教育双减)&#xff0c;所以对es的了解其实仅仅是些皮毛&#xff0c;当然elk在我的任职…

稀疏数组搜索

题目链接 稀疏数组搜索 题目描述 注意点 字符串数组中散布着一些空字符串words的长度在[1, 1000000]之间字符串数组是排好序的数组中的字符串不重复 解答思路 因为数组中的字符串是排好序的&#xff0c;所以首先想到的是二分查找&#xff0c;先将数组中长度与s相同的字符串…

全网都在疯传的最新蓝海风口项目!

最近全网都在疯传这种视频&#xff0c;想必兄弟们都见到过了&#xff01; 大家看这个号&#xff0c;1天的时间&#xff0c;2个作品&#xff0c;第2个直接就爆了&#xff0c;昨天看点赞还是3.8w&#xff0c;今天已经10w了&#xff0c;这是妥妥的风口啊&#xff01; 大家有没有想…

io_contextttttttttttt

创建上下文——io_context_t 它是一个上下文结构&#xff0c;在内部它包含一个完成队列&#xff0c;在线程之间是可以共享的。 提交请求——iocb io回调数据结构&#xff0c;和io_submit配合使用。 处理结果 通过io_event处理结果&#xff0c; struct io_event {void *data…

TIS人人都会用的数据集成产品

文章目录 1.TIS是什么&#xff1f;1.1 简介1.2 官网及项目地址1.3 架构 2.功能特性2.1 基于Web UI的开箱即用2.2 支持分布式任务分发2.3 全新的基于微内核的运行环境2.4 功能覆盖DataX大部分&#xff08;Reader/Writer&#xff09;Plugin2.5 重构DataX的Classloader2.6 支持RDB…

科比老大职业生涯数据预测(基于随机森林模型)

1.实验背景 科比布莱恩特&#xff0c;作为NBA历史上最伟大的篮球运动员之一&#xff0c;他的职业生涯充满了无数精彩瞬间。 科比于1996年以13顺位的选秀身份进入联盟&#xff0c;一生都效力于洛杉矶湖人队。于2016年宣布退役&#xff0c;职业生涯获奖无数&#xff0c;5次NBA总…

99. 岛屿数量

题目描述&#xff1a;给定一个由 1&#xff08;陆地&#xff09;和 0&#xff08;水&#xff09;组成的矩阵&#xff0c;你需要计算岛屿的数量。岛屿由水平方向或垂直方向上相邻的陆地连接而成&#xff0c;并且四周都是水域。你可以假设矩阵外均被水包围。 输入描述&#xff1a…

EXTI寄存器,AFIO的简洁,EXTI配置的流程

一&#xff0c;AFIO简介 AFIO是Alternate Function Input/Output 的缩写&#xff0c;表示复用功能IO&#xff0c;主要用于实现IO端口的复用功能以及外部中断的控制 STM32外设有很多I/O以及内置外设&#xff08;如12C&#xff0c;ADC,ISP,USART等&#xff09;。为节省引出管脚的…

命令行运行git reflog(reference log)报错的解决办法

文章目录 1. 检查 Git 是否已安装2. 检查 PATH 环境变量3. 重新安装 Git 在Git中&#xff0c; reflog的英文全称是 “ reference log”。意思是 引用日志&#xff08;参考日志&#xff09;。它记录了本地仓库中HEAD和分支引用所指向的提交的变更历史。这包括了你所有的提交&…

经典低功耗四通道运算放大器LM324

前言&#xff1a; SOP14封装LM324 这个LM324运放有几十年的历史了吧&#xff1f;很普通&#xff0c;很常用&#xff0c;搞电路的避免不了接触运放&#xff0c;怎么选择运放&#xff0c;是工程师关心的问题吧&#xff1f; 从本文开始&#xff0c;将陆续发一些常用的运放&#xf…

【AI学习】无线AI的问题和挑战

无线AI&#xff0c;即无线人工智能&#xff0c;是指内生于未来&#xff08;6G&#xff09;无线通信系统并通过无线架构、无线数据、无线算法和无线应用所呈现出来的新的人工智能技术体系。 最近一直在进行无线AI的调研&#xff0c;感觉真的是路漫漫其修远兮。业界有一些探索&a…

【人工智能】--生成对抗网络

个人主页&#xff1a;欢迎来到 Papicatch的博客 课设专栏 &#xff1a;学生成绩管理系统 专业知识专栏&#xff1a; 专业知识 文章目录 &#x1f349;引言 &#x1f349;GAN 的基本原理 &#x1f348;生成器&#xff08;Generator&#xff09; &#x1f348;判别器&…

【前端知识】一篇速成 建议收藏

HTML基础概念 正式敲代码之前呢,我们先来看几个概念: 0 静态网页和动态网页 静态网页: 页面的内容和显示效果就基本上不会发生变化了--除非你修改页面代码。 动态网页: 页面代码虽然没有变&#xff0c;但是显示的内容却是可以随着时间、环境或者数据库操作的结果而发生改变的…

解决SeaTunnel 2.3.4版本写入S3文件报错问题

在使用Apache SeaTunnel时&#xff0c;我遇到了一个写入S3文件的报错问题。通过深入调试和分析&#xff0c;找到了问题所在&#xff0c;并提出了相应的解决方案。 本文将详细介绍报错情况、参考资料、解决思路以及后续研究方向&#xff0c;希望对大家有帮助&#xff01; 一、…

PyTorch - 神经网络基础

神经网络的主要原理包括一组基本元素&#xff0c;即人工神经元或感知器。它包括几个基本输入&#xff0c;例如 x1、x2… xn &#xff0c;如果总和大于激活电位&#xff0c;则会产生二进制输出。 样本神经元的示意图如下所述。 产生的输出可以被认为是具有激活电位或偏差的加权…

Java通过GeoLite2-City.mmdb 进行IP信息查询地理定位和经纬度筛选。

引入依赖 <dependency><groupId>com.maxmind.geoip2</groupId><artifactId>geoip2</artifactId><version>4.2.0</version> </dependency>下载数据文件&#xff1a;https://download.lin2ur.cn/GeoLite2/ package com.cqclo…

经典递归分析

在前面一篇中, 已经看过许多直观的递归的例子, 在这篇里, 将分析两个经典的递归问题, 阶乘与菲波那契数列数列, 在此过程中, 还将对比递归与循环(迭代)间的异同, 探讨递归与内存中的栈的关系, 以及递归的效率等问题. 如无特别说明, 示例使用的是 Java, IDE 则为 Eclipse. 阶乘(…

Matplotlib 简介

import matplotlib.pyplot as plt plt.plot([1, 2, 3, 4]) plt.ylabel(some numbers) plt.show() 当使用plot只传入单个数组时&#xff0c;matplotlib会认为这是y的值&#xff0c;并自动生成长度相同&#xff0c;但是从0开始的x值&#xff0c;所以这里的x会自动生成为 [0,1,2,…

python自动化办公之BeautifulSoup爬取并解析html文本

用到的库&#xff1a;BeautifulSoup 实现效果&#xff1a;爬取网站内容&#xff0c;拿到html文本并解析html文本 代码&#xff1a; 先爬取 # 先导入requests包 import requests urlhttps://www.baidu.com responserequests.get(url) # 做1个断言&#xff0c;如果执行成功&a…
最新文章