Java的注解和注解解析器是什么,运行方法如何,架构师必会知识体系

news2025/1/9 17:56:52

文章目录

    • 什么是[元数据](https://so.csdn.net/so/search?q=元数据&spm=1001.2101.3001.7020)(metadata)
    • 什么是注解(Annotation)?
    • 元注解
    • 自定义注解
    • 注解处理器

什么是元数据(metadata)

​ 元数据由metadata译来,所谓的元数据就是“关于数据的数据”,更通俗的说就是描述数据的数据,对数据及信息资源的描述性信息.比如说一个文本文件,有创建时间,创建人,文件大小等数据,这都可以理解为是元数据.

​ 在java中,元数据以标签的形式存在java代码中,它的存在并不影响程序代码的编译和执行,通常它被用来生成其它的文件或运行时知道被运行代码的描述信息。java当中的javadoc和注解都属于元数据.

什么是注解(Annotation)?

​ 注解是从java 5.0开始加入,可以用于标注包,类,方法,变量等.比如我们常见的@Override,再或者Android源码中的@hide,@systemApi,@privateApi等

对于@Override,多数人往往都是知其然而不知其所以然,今天我就来聊聊Annotation背后的秘密,开始正文.

元注解

元注解就是定义注解的注解,是java提供给我们用于定义注解的基本注解.在java.lang.annotation包中我们可以看到目前元注解共有以下几个:

  1. @Retention
  2. @Target
  3. @Inherited
  4. @Documented
  5. @interface

下面我们将集合@Override注解来解释着5个基本注解的用法.

@interface

@interface是java中用于声明注解类的关键字.使用该注解表示将自动继承java.lang.annotation.Annotation类,该过程交给编译器完成.

因此我们想要定义一个注解只需要如下做即可,以@Override注解为例

public @interface Override {}
需要注意:在定义注解时,不能继承其他注解或接口

@Retention

@Retention:该注解用于定义注解保留策略,即定义的注解类在什么时候存在(源码阶段 or 编译后 or 运行阶段).该注解接受以下几个参数:RetentionPolicy.SOURCE,RetentionPolicy.CLASS,RetentionPolicy.RUNTIME,其具体使用及含义如下:

img

来看一下@Override注解的保留策略:

@Retention(RetentionPolicy.SOURCE)public @interface Override {}

这表明@Override注解只在源码阶段存在,javac在编译过程中去去掉该注解.

@Target

该注解用于定义注解的作用目标,即注解可以用在什么地方,比如是用于方法上还是用于字段上,该注解接受以下参数:

img

以@Override为例,不难看出其作用目标为方法:

@Target(ElementType.METHOD)public @interface Override {}

到现在,通过@interface,@Retention,@Target已经可以完整的定义一个注解,来看@Override完整定义:

@Target(ElementType.METHOD)@Retention(RetentionPolicy.SOURCE)public @interface Override {}

@Inherited

默认情况下,我们自定义的注解用在父类上不会被子类所继承.如果想让子类也继承父类的注解,即注解在子类也生效,需要在自定义注解时设置@Inherited.一般情况下该注解用的比较少.

@Documented

该注解用于描述其它类型的annotation应该被javadoc文档化,出现在api doc中.

比如使用该注解的@Target会出出现在api说明中.

@Documented@Retention(RetentionPolicy.RUNTIME)@Target(ElementType.ANNOTATION_TYPE)public @interface Target {

    ElementType[] value();}
借助@Interface,@Target,@Retention,@Inherited,@Documented这五个元注解,我们就可以自定义注解了,其中前三个注解是任何一个注解都必备具备的.

自定义注解

格式:

public @interface 注解名 {定义体}

定义体就是方法的集合,每个方法实则是声明了一个配置参数.方法的名称作为配置参数的名称,方法的返回值类型就是配置参数的类型.和普通的方法不一样,可以通过default关键字来声明配置参数的默认值.

需要注意:

此处只能使用public或者默认的defalt两个权限修饰符配置参数的类型只能使用基本类型(byte,boolean,char,short,int,long,float,double)和String,Enum,Class,annotation对于只含有一个配置参数的注解,参数名建议设置中value,即方法名为value配置参数一旦设置,其参数值必须有确定的值,要不在使用注解的时候指定,要不在定义注解的时候使用default为其设置默认值,对于非基本类型的参数值来说,其不能为null.

像@Override这样,没有成员定义的注解称之为标记注解.

注解处理器

上面我们已经学会了如何定义注解,要想注解发挥实际作用,需要我们为注解编写相应的注解处理器.根据注解的特性,注解处理器可以分为运行时注解处理和编译时注解处理器.运行时处理器需要借助反射机制实现,而编译时处理器则需要借助APT来实现.

无论是运行时注解处理器还是编译时注解处理器,主要工作都是读取注解及处理特定注解,从这个角度来看注解处理器还是非常容易理解的.

​ 注解处理器是(Annotation Processor)是javac的一个工具,用来在编译时扫描和编译和处理注解(Annotation)。你可以自己定义注解和注解处理器去搞一些事情。一个注解处理器它以Java代码或者(编译过的字节码)作为输入,生成文件(通常是java文件)。这些生成的java文件不能修改,并且会同其手动编写的java代码一样会被javac编译。看到这里加上之前理解,应该明白大概的过程了,就是把标记了注解的类,变量等作为输入内容,经过注解处理器处理,生成想要生成的java代码。

运行时注解处理器(不建议使用)

熟悉java反射机制的同学一定对java.lang.reflect包非常熟悉,该包中的所有api都支持读取运行时Annotation的能力,即属性为@Retention(RetentionPolicy.RUNTIME)的注解.

在java.lang.reflect中的AnnotatedElement接口是所有程序元素的(Class,Method)父接口,我们可以通过反射获取到某个类的AnnotatedElement对象,进而可以通过该对象提供的方法访问Annotation信息,常用的方法如下:

img

​ 运行时注解处理器的编写本质上就是通过反射获取注解信息,随后进行其他操作。编译一个运行时注解处理器就是这么简单。运行时注解通常多用于参数配置类模块。

编译时注解处理器

不同于运行时注解处理器,编写编译时注解处理器(Annotation Processor Tool).

APT用于在编译时期扫描和处理注解信息.一个特定的注解处理器可以以java源码文件或编译后的class文件作为输入,然后输出另一些文件,可以是.java文件,也可以是.class文件,但通常我们输出的是.java文件.(注意:并不是对源文件修改).如果输出的是.java文件,这些.java文件回合其他源码文件一起被javac编译.

你可能很纳闷,注解处理器是到底是在什么阶段介入的呢?好吧,其实是在javac开始编译之前,这也就是通常我们为什么愿意输出.java文件的原因.

注解最早是在java 5引入,主要包含apt和com.sum.mirror包中相关mirror api,此时apt和javac是各自独立的。从java 6开始,注解处理器正式标准化,apt工具也被直接集成在javac当中。

我们还是回到如何编写编译时注解处理器这个话题上,编译一个编译时注解处理主要分两步:

1、继承AbstractProcessor,实现自己的注解处理器

2、注册处理器,并打成jar包

首先来看一下一个标准的注解处理器的格式:

public class MyAnnotationProcessor extends AbstractProcessor {
    @Override    public Set<String> getSupportedAnnotationTypes() {        return super.getSupportedAnnotationTypes();    }
    @Override    public SourceVersion getSupportedSourceVersion() {        return super.getSupportedSourceVersion();    }
    @Override    public synchronized void init(ProcessingEnvironment processingEnv) {        super.init(processingEnv);    }
    @Override    public boolean process(Set<? extends TypeElement> annotations, RoundEnvironment roundEnv) {        return false;    }}

img

​ 编写一个注解处理器首先要对ProcessingEnvironment和RoundEnvironment非常熟悉。接下来我们一览这两个类的风采.首先来看一下ProcessingEnvironment类:

public interface ProcessingEnvironment {
    Map<String,String> getOptions();
    //Messager用来报告错误,警告和其他提示信息    Messager getMessager();
    //Filter用来创建新的源文件,class文件以及辅助文件    Filer getFiler();
    //Elements中包含用于操作Element的工具方法    Elements getElementUtils();
     //Types中包含用于操作TypeMirror的工具方法    Types getTypeUtils();
    SourceVersion getSourceVersion();
    Locale getLocale();}

Element

element表示一个静态的,语言级别的构件。而任何一个结构化文档都可以看作是由不同的element组成的结构体,比如XML,JSON等。

对于java源文件来说,Element代表程序元素:包,类,方法都是一种程序元素,他同样是一种结构化文档:

package com.closedevice;             //PackageElementpublic class Main{                  //TypeElement    private int x;                  //VariableElement    private Main(){                 //ExecuteableElement    }    private void print(String msg){ //其中的参数部分String msg为TypeElement    }}

img

img

TypeMirror

这三个类也需要我们重点掌握:

DeclaredType代表声明类型:类类型还是接口类型,当然也包括参数化类型,比如Set,也包括原始类型

TypeElement代表类或接口元素,而DeclaredType代表类类型或接口类型。

TypeMirror代表java语言中的类型.Types包括基本类型,声明类型(类类型和接口类型),数组,类型变量和空类型。也代表通配类型参数,可执行文件的签名和返回类型等。TypeMirror类中最重要的是getKind()方法,该方法返回TypeKind类型,为了方便大家理解,这里附上其源码:

public enum TypeKind {    BOOLEAN,BYTE,SHORT,INT,LONG,CHAR,FLOAT,DOUBLE,VOID,NONE,NULL,ARRAY,DECLARED,ERROR,  TYPEVAR,WILDCARD,PACKAGE,EXECUTABLE,OTHER,UNION,INTERSECTION;    public boolean isPrimitive() {        switch(this) {        case BOOLEAN:        case BYTE:        case SHORT:        case INT:        case LONG:        case CHAR:        case FLOAT:        case DOUBLE:            return true;        default:            return false;        }    }}

​ 简单来说,Element代表源代码,TypeElement代表的是源码中的类型元素,比如类。虽然我们可以从TypeElement中获取类名,TypeElement中不包含类本身的信息,比如它的父类,要想获取这信息需要借助TypeMirror,可以通过Element中的asType()获取元素对应的TypeMirror。

然后来看一下RoundEnvironment,这个类比较简单,一笔带过:

public interface RoundEnvironment {    boolean processingOver();     //上一轮注解处理器是否产生错误    boolean errorRaised();     //返回上一轮注解处理器生成的根元素    Set<? extends Element> getRootElements();   //返回包含指定注解类型的元素的集合    Set<? extends Element> getElementsAnnotatedWith(TypeElement a);    //返回包含指定注解类型的元素的集合    Set<? extends Element> getElementsAnnotatedWith(Class<? extends Annotation> a);}

Filer

​ Filer用于注解处理器中创建新文件,由于Filer用起来实在比较麻烦,后面我们会使用javapoet简化我们的操作.

打包注解处理器的时候需要一个特殊的文件 javax.annotation.processing.Processor 在 META-INF/services 路径下

img

新建项目必要配置:

 //javapoet代码生成框架 implementation  'com.squareup:javapoet:1.8.0'  //注解处理器 implementation 'com.google.auto.service:auto-service:1.0-rc6' annotationProcessor  'com.google.auto.service:auto-service:1.0-rc6'

编译时注解demo示例地址:https://gitee.com/yutg/apt.git

项目结构

--apt-demo----bindview-annotation(Java Library)//注解定义----bindview-api(Android Library)//定义SDK接口方法----bindview-compiler(Java Library)//注解处理器相关操作及生成java文件----app(Android App)

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

网络篇汇总

路由器&#xff1a;属于网关设备&#xff0c;通过路由器可以将各种局域网、城域网、广域网连接起来&#xff0c;一般工作于网络层。它会根据信号的情况自动选择和设定路由&#xff0c;以最佳路径&#xff0c;按照前后顺序发送信号。路由器可连接多个逻辑上分开的网络&#xff0…

2023最新SSM计算机毕业设计选题大全(附源码+LW)之java随骑共享系统m7i8o

要对当前自己的学校对于计算机毕业设计的要求以及严格程度有所了解&#xff0c;这个主要是借助上一届已经毕业的学长学姐了解一下&#xff0c;但是有一点要注意的是&#xff0c;对于每一届毕业生的毕业设计的处理&#xff0c;学校都有很大的调整&#xff0c;这一点尤其重要。其…

静态HTML个人博客主页 简单个人网页设计作业 DW个人网站模板下载 大学生简单个人网页作品代码 个人网页制作 学生个人网页设计作业

&#x1f389;精彩专栏推荐&#x1f447;&#x1f3fb;&#x1f447;&#x1f3fb;&#x1f447;&#x1f3fb; ✍️ 作者简介: 一个热爱把逻辑思维转变为代码的技术博主 &#x1f482; 作者主页: 【主页——&#x1f680;获取更多优质源码】 &#x1f393; web前端期末大作业…

力扣(LeetCode)139. 单词拆分(C++)

字符串哈希&动态规划 一眼暴力&#xff0c;发现根本做不了&#xff0c;想着想着&#xff0c;发现 dpdpdp 很好想&#xff0c;来试试。 枚举字符串 sss 的所有位置作为起始点&#xff0c;如图状态转移。 规定 : f[0]f[0]f[0] 表示空字符的集合&#xff0c;可以用空串组成…

【数据结构与算法】模拟实现LinkedList类

文章目录LinkedList简介头插法创建链表尾插法创建链表任意位置插入,第一个数据节点为0号下标查找是否包含关键字key是否在链表当中删除第一次出现关键字为key的节点删除所有值为key的节点得到链表的长度打印链表清空链表完整代码&#xff1a;总结:LinkedList简介 Java LinkedLi…

SAP PS 第6节 项目产成品产出

SAP PS 第6节 项目产成品产出及差异处理1 模拟场景说明1.1 拖拽负库存1.2 发料原材料及报工1.3 执行副产品入库migo发预留1.4 CNS0交货1.5 后面开票产生收入按照项目结算即可项目上有一类比较另类的玩法&#xff0c;就是舍弃PP&#xff0c;依靠网络活动的负库存&#xff0c;实现…

Android Edittext密码类型显示字符串修改实现

Android Edittext密码类型显示字符串修改实现 文章目录Android Edittext密码类型显示字符串修改实现一、前言&#xff1a;二、效果三、实现1、系统级设置2、应用级设置3、单个EditText设置&#xff08;1&#xff09;自定义显示符合类&#xff08;2&#xff09;EditText使用自定…

认识一下 Kubernetes 多集群服务 API

由于各种原因&#xff0c;采用 Kubernetes 的企业内部存在着几个、几十甚至上百个集群。比如处于研发流程上的考虑&#xff0c;不同环境下都存在独立的集群&#xff1b;监管层面的考虑&#xff0c;就地存储的用户数据需要搭配应用集群&#xff1b;单个集群的容量限制&#xff0…

(四)Redis的持久化

一 什么是redis持久化 因为Redis数据是基于内存读写,为防止Redis服务器关闭或者宕机造成数据的丢失,我们通常需要对redis做持久化,即:把内存中的数据(命令)保存一份到磁盘中来做一份备份,当redis服务关闭或宕机后,在Redis服务器重启后将数据从磁盘加载到内存中,不至于造成数据…

一招教你轻松使用公网远程访问公司内网

企业远程访问需求 众多企业都会在总部搭建各类项目管理办公系统&#xff08;如OA、ERP、CRM、财务系统等等&#xff09;&#xff0c;以提高员工的办公及管理效率。 不少出差在外或者居家办公的员工需要从外部网络访问内网来登录各类系统&#xff0c;以满足办公协作管理的需…

电容笔和触控笔有啥区别?双十二质量好的电容笔推荐

从导电材料、作用机理、用途等方面&#xff0c;电容笔与普通触控笔相比有很大的不同。电容笔的笔尖尺寸适中&#xff0c;笔尖材质一般比较耐用。随着科技的进步&#xff0c;人们的生活水准不断提高&#xff0c;无论是绘制图纸&#xff0c;或是会议纪要&#xff0c;都需要一款更…

(附源码)ssm教学成绩管理系统 毕业设计 282029

教学成绩管理系统的设计与实现 摘 要 随着互联网趋势的到来&#xff0c;各行各业都在考虑利用互联网将自己推广出去&#xff0c;最好方式就是建立自己的互联网系统&#xff0c;并对其进行维护和管理。在现实运用中&#xff0c;应用软件的工作规则和开发步骤&#xff0c;采用Jav…

大学生圣诞网页设计制作成品 圣诞节静态HTML网页作业作品 简单DIV CSS布局网站

&#x1f389;精彩专栏推荐 &#x1f4ad;文末获取联系 ✍️ 作者简介: 一个热爱把逻辑思维转变为代码的技术博主 &#x1f482; 作者主页: 【主页——&#x1f680;获取更多优质源码】 &#x1f393; web前端期末大作业&#xff1a; 【&#x1f4da;毕设项目精品实战案例 (10…

树的企业应用-哈夫曼编码树-有趣的数据压缩算法

树的企业应用-哈夫曼编码树-有趣的数据压缩算法 哈夫曼编码 描述 张三去李四家里,但 李四是一个女生,所以张三找李四去上海迪尼斯玩 … 亚历山大.张三去伊丽莎白.李四家里,但 伊丽莎白.李四是一个女生,所以亚历山大.张三找伊丽莎白.李四去美国迪尼斯玩 … 我们发现 一个关键…

C++中STL用法超详细总结(收藏级)

1 什么是STL&#xff1f; STL&#xff08;Standard Template Library&#xff09;&#xff0c;即标准模板库&#xff0c;是一个具有工业强度的&#xff0c;高效的C程序库。它被容纳于C标准程序库&#xff08;C Standard Library&#xff09;中&#xff0c;是ANSI/ISO C标准中最…

密码学(1)RSA与AES算法原理

什么是RSA 在1977年&#xff0c;Ron Rivest, Adi Shami和Leonard Adleman这三个人开发了一个新的算法&#xff0c;并用他们三个名字的首字母来命名这个算法&#xff0c;这个算法名叫RSA 非对称加密与对称加密 RSA算法采用的是非对称加密&#xff0c;假如我有你的公钥&#x…

从零到一手写一个小型RPC框架——介绍篇

RPC框架介绍 RPC框架是微服务的通信工具&#xff0c;其涉及到网络传输、服务注册、序列化、代理模式等等知识的学习与使用&#xff0c;是微服务的入门框架。 现有的比较知名的RPC框架有阿里巴巴开源的Dubbo&#xff0c;谷歌的开源RPC框架gRPC&#xff0c;甚至SpringCloud中所…

vue2笔记4(服务代理、插槽、vuex、路由)

vue脚手架 ## vue脚手架配置代理 vue在请求服务器的资源的时候&#xff0c;自身使用的是8080端口&#xff0c;如果服务器端口号是别的&#xff0c;这时候直接访问的话就会出现跨域的问题无法访问&#xff0c;就需要使用一个代理来访问服务器。vue配置脚手架的代理有两种方式。…

Unreal Engine学习

1&#xff0c;什么是组件&#xff1f; 继承于Uobject的为了给actor以及其他在场景中显示出来的类增加新的功能的一个东西 2&#xff0c;可以自己创建一个蓝图&#xff0c;然后将蓝图拖到场景中&#xff0c;也可以直接在c class中将蓝图拖到场景中 3&#xff0c;创建一个组件&…

vuex的新写法引入mapState省略$store.state

vuex的新写法&#xff1a; state简写&#xff08;映射&#xff09;&#xff1a;mapState state里都是状态&#xff0c;所以mapState在computed中使用&#xff1b;state映射也就是引入mapState&#xff0c;然后state可以简写$store.state.状态 > 状态&#xff1a;前面的$s…