android注解之APT和javapoet

前言

前面我们已经讲过注解的基本知识,对于注解还不太了解的,可以去看一下之前的文章,

android 注解详解_袁震的博客-CSDN博客。

之前我们在讲注解的时候,提到过APT和JavaPoet,那么什么是APT和JavaPoet呢?下面我们来详细讲解一下。

1,APT是什么?

APT,英文全称Annotation Processing Tool,是一种处理注释的工具,它对源代码文件进行检测找出其中的Annotation,根据注解自动生成代码, 如果想要自定义的注解处理器能够正常运行,必须要通过APT工具来进行处理。 也可以这样理解,只有通过声明APT工具后,程序在编译期间自定义注解解释器才能执行。注意,是在编译期间

简单来讲,就是根据我们定义的注释规则,帮助我们生成代码,生成类文件。

2,APT中的元素

在APT中,它会分为包元素,类元素,属性元素,方法元素。那么,这些元素的意义是什么呢?

 首先,我们需要明白APT真正的作用是什么。就拿ButterKnife来说,他真正要实现的就是我们通过BindView,把id传给注解,然后就会在编译时动态生成很多类,专门去处理你绑定的这些id,从而达到你只需要几行代码就能实现绑定,点击事件等功能。那为什么不能直接就写好类去处理,而非要到编译期去自动生成类处理呢?因为它不知道你会传哪些id,所以需要动态的生成。

所以,我认为APT的主要作用就是帮你动态生成类。

import androidx.appcompat.app.AppCompatActivity;//PackageElement 包元素/节点public class MainActivity2 extends AppCompatActivity { // TypeElement 类元素/节点private int a;// VariableElement 属性元素/节点@Overrideprotected void onCreate(@Nullable Bundle savedInstanceState) {// ExecuteableElement 方法元素/节点super.onCreate(savedInstanceState);}
}

这些元素的意义就是,它们会提供相关信息,来帮助你后面生成类。

包元素
PackageElement
表示一个包程序元素。提供对有关包及其成员的信息的访问
方法元素
ExecutableElement
表示某个类或接口的方法、构造方法或初始化程序(静态或实例)
类元素
TypeElement
表示一个类或接口程序元素。提供对有关类型及其成员的信息的访问
属性元素
VariableElement
表示一个字段、enum 常量、方法或构造方法参数、局部变量或异常参数

3,APT中常用的API

在AbstractProcessor中,有两个方法是核心方法:

    
//初始化工作,主要做一些准备工作
public synchronized void init(ProcessingEnvironment processingEnv) {}//处理注解 核心方法
//annotations 使用了支持处理注解的节点集合
//roundEnv 当前或是之前的运行环境,可以通过该对象查找的注解
//return true 表示后续处理器不会再处理(已经处理完成)
public abstract boolean process(Set<? extends TypeElement> annotations,RoundEnvironment roundEnv);

上面ProcessingEnvironment中常用的api如下:

getElementUtils()获取操作Element的工具类
getMessager()获取Messager,用来打印日志相关信息
getFiler()获取文件生成器,类等最终要生成的文件,都是通过生成器生成的
getTypeUtils()获取类信息的工具类,用于操作TypeMirror的工具方法
getOptions()主要用来接收应用传过来的数据

上面RoundEnvironment常用api如下:

//获取所有被@YuanZhen注解的元素集合
Set<? extends Element> elements = roundEnvironment.getElementsAnnotatedWith(YuanZhen.class);

Element常用api如下:

getEnclosedElements()
返回该元素直接包含的子元素
getEnclosingElement()
返回包含该element的父element,与上一个方法相反
getKind()
返回element的类型,判断是哪种element
getModifiers()
获取修饰关键字,入public static final等关键字
getSimpleName()
获取名字,不带包名
getQualifiedName()
获取全名,如果是类的话,包含完整的包名路径
getParameters()
获取方法的参数元素,每个元素是一个VariableElement
getReturnType()
获取方法元素的返回值
getConstantValue()
如果属性变量被final修饰,则可以使用该方法获取它的值

4,APT环境搭建

4.1新建注解javalib,命名为compiler

4.2在compiler的buidl.gradle中添加依赖,用来注册注解处理器

compileOnly'com.google.auto.service:auto-service:1.0-rc4'
annotationProcessor'com.google.auto.service:auto-service:1.0-rc4'

4.3新建AnnotationProcessor

新建MyAnnotationProcessor类,继承AnnotationProcessor

// AutoService则是固定的写法,加个注解即可
// 通过auto-service中的@AutoService可以自动生成AutoService注解处理器,用来注册
// 用来生成 META-INF/services/javax.annotation.processing.Processor 文件
@AutoService(Processor.class)// 允许/支持的注解类型,让注解处理器处理
@SupportedAnnotationTypes({"com.yuanzhen.yuanzhenannotation.YuanZhen"})// 指定JDK编译版本
@SupportedSourceVersion(SourceVersion.RELEASE_7)
public class MyAnnotationProcessor extends AbstractProcessor {@Overridepublic synchronized void init(ProcessingEnvironment processingEnv) {super.init(processingEnv);}@Overridepublic boolean process(Set<? extends TypeElement> set, RoundEnvironment       roundEnvironment) {return false;}
}

4.4新建注解javalib  命名为yuanzhenannotation

4.5创建自己的注解

@Target(ElementType.TYPE) //作用与类上
@Retention(RetentionPolicy.SOURCE) //在编译时期生效
public @interface YuanZhen {String value();//一个默认值
}

4.6添加依赖

在compiler中依赖yuanzhenannotation

在app中添加依赖:

4.7在app中使用注解:

@YuanZhen("study")
public class MainActivity extends AppCompatActivity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);}
}

至此,APT环境搭建完成

5,APT获取注解

在MyAnnotationProcessor类中:

// AutoService则是固定的写法,加个注解即可
// 通过auto-service中的@AutoService可以自动生成AutoService注解处理器,用来注册
// 用来生成 META-INF/services/javax.annotation.processing.Processor 文件
@AutoService(Processor.class)// 允许/支持的注解类型,让注解处理器处理
@SupportedAnnotationTypes({"com.yuanzhen.yuanzhenannotation.YuanZhen"})// 指定JDK编译版本
@SupportedSourceVersion(SourceVersion.RELEASE_7)
public class MyAnnotationProcessor extends AbstractProcessor {private Messager messager;// 用来打印日志相关信息private Elements elementUtils;// 操作Element的工具类(类,函数,属性,其实都是Element)private Filer filer;//文件生成器, 类 资源 等,就是最终要生成的文件 是需要Filer来完成的private Types typeUtils;// type(类信息)的工具类,包含用于操作TypeMirror的工具方法@Overridepublic synchronized void init(ProcessingEnvironment processingEnv) {super.init(processingEnv);messager = processingEnv.getMessager();elementUtils = processingEnv.getElementUtils();filer = processingEnv.getFiler();typeUtils = processingEnv.getTypeUtils();}@Overridepublic boolean process(Set<? extends TypeElement> set, RoundEnvironment roundEnvironment) {//因为javalib没有Log,所以我们使用messager来打印messager.printMessage(Diagnostic.Kind.NOTE,"aaaaaaZZZZZ");//打印 //获取所有被@YuanZhen注解的元素集合Set<? extends Element> elements = roundEnvironment.getElementsAnnotatedWith(YuanZhen.class);for (Element element : elements) {String className =element.getSimpleName().toString();//获取元素名messager.printMessage(Diagnostic.Kind.NOTE,"-----="+className);//打印类名YuanZhen annotation = element.getAnnotation(YuanZhen.class);//获取注解messager.printMessage(Diagnostic.Kind.NOTE,"-----value="+annotation.value());//打印注解参数}return true;}
}

具体的api在上文已经有个介绍,下面看Build日志输出:

参数study已经传递过来,类名MainActivity也已经获取到。

6,传统的生成类的方式

上面已经完成了APT环境的配置,下面就是动态生成类了,在javapoet之前,传统的生成类的方式就是采用字符串拼接的样式。最典型的应用就是EventBus。

private void createInfoIndexFile(String index) {BufferedWriter writer = null;try {// 通过注解处理的文件操作工具类创建源文件JavaFileObject sourceFile = processingEnv.getFiler().createSourceFile(index);int period = index.lastIndexOf('.');// 截取包名和类名String myPackage = period > 0 ? index.substring(0, period) : null;String clazz = index.substring(period + 1);writer = new BufferedWriter(sourceFile.openWriter());// 以下就是写入生成的源文件中的代码if (myPackage != null) {writer.write("package " + myPackage + ";\n\n");}writer.write("import org.greenrobot.eventbus.meta.SimpleSubscriberInfo;\n");writer.write("import org.greenrobot.eventbus.meta.SubscriberMethodInfo;\n");writer.write("import org.greenrobot.eventbus.meta.SubscriberInfo;\n");writer.write("import org.greenrobot.eventbus.meta.SubscriberInfoIndex;\n\n");writer.write("import org.greenrobot.eventbus.ThreadMode;\n\n");writer.write("import java.util.HashMap;\n");writer.write("import java.util.Map;\n\n");writer.write("/** This class is generated by EventBus, do not edit. */\n");writer.write("public class " + clazz + " implements SubscriberInfoIndex {\n");writer.write("    private static final Map<Class<?>, SubscriberInfo> SUBSCRIBER_INDEX;\n\n");writer.write("    static {\n");writer.write("        SUBSCRIBER_INDEX = new HashMap<Class<?>, SubscriberInfo>();\n\n");// 写入订阅方法相关信息writeIndexLines(writer, myPackage);writer.write("    }\n\n");writer.write("    private static void putIndex(SubscriberInfo info) {\n");writer.write("        SUBSCRIBER_INDEX.put(info.getSubscriberClass(), info);\n");writer.write("    }\n\n");writer.write("    @Override\n");writer.write("    public SubscriberInfo getSubscriberInfo(Class<?> subscriberClass) {\n");writer.write("        SubscriberInfo info = SUBSCRIBER_INDEX.get(subscriberClass);\n");writer.write("        if (info != null) {\n");writer.write("            return info;\n");writer.write("        } else {\n");writer.write("            return null;\n");writer.write("        }\n");writer.write("    }\n");writer.write("}\n");} catch (IOException e) {throw new RuntimeException("Could not write source for " + index, e);} finally {if (writer != null) {try {writer.close();} catch (IOException e) {//Silent}}}
}

通过上面可以看到,Eventbus是通过字符串拼接的形式来生成类的,这种方式虽然比较简单,但是也不是一件轻松的体力活,不符合OOP的编程思想。下面我们就来介绍一下现在最流行的写法,采用javapoet。

7,JavaPoet简介

JavaPoet是square推出的开源java代码生成框架,提供Java Api生成.java源文件,这个框架功能非常实用,也是我们习惯的Java面向对象OOP语法, 可以很方便的使用它根据注解生成对应代码。 通过这种自动化生成代码的方式, 可以让我们用更加简洁优雅的方式要替代繁琐冗杂的重复工作。

8,JavaPoet中常用的类

MethodSpec代表一个构造函数或方法声明
TypeSpec代表一个类,接口,或者枚举声明
FieldSpec代表一个成员变量,一个字段声明
JavaFile包含一个顶级类的Java文件
ParameterSpec用来创建参数
AnnotationSpec用来创建注解
ClassName用来包装一个类
TypeName类型,如在添加返回值类型是使用 TypeName.VOI

$S 字符串,如:$S, ”hello

$T 类、接口,如:$T, MainActivit

9,JavaPoet生成类的思想

传统模式生成类的思想是先写包,然后类,然后方法。

JavaPoet的思想恰恰相反,它是先方法,然后类,然后包。

10,JavaPoet 使用

10.1 导包

在complier的build.gradle里面导包

 implementation "com.squareup:javapoet:1.9.0"

10.2写出要生成的class

/**
package com.yuanzhen.yuanzhenannotation;public class MyClass {public static void main(String[] args) {System.out.println("Hello, yuanzhen");}
}
*/

10.3 通过javapoet生成该类

 // 1.方法 
MethodSpec mainMethod = MethodSpec.methodBuilder("main")//添加方法名.addModifiers(Modifier.PUBLIC, Modifier.STATIC)//添加修饰符.returns(void.class)//添加返回值.addParameter(String[].class, "args")//添加方法参数.addStatement("$T.out.println($S)", System.class, "Hello, YuanZhen!")//添加内容.build();// 2.类TypeSpec helloWorld = TypeSpec.classBuilder("MyClass1")//添加类名.addModifiers(Modifier.PUBLIC, Modifier.FINAL)//添加修饰符.addMethod(mainMethod)//添加方法.build();// 3.包JavaFile packagef = JavaFile.builder("com.yuanzhen.apt1", helloWorld).build();// 去生成try {packagef.writeTo(filer);messager.printMessage(Diagnostic.Kind.NOTE, "success...");} catch (IOException e) {e.printStackTrace();messager.printMessage(Diagnostic.Kind.NOTE, "error...");}

10.4在app build中查看生成的文件

11,总结

关于APT和JavaPoet的使用就讲完了,利用这个技术,我们可以实现很多强大的功能。

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

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

相关文章

【Mysql】Mysql获取排班时间段中的休息时间段方法

在MySQL中&#xff0c;可以使用自连接&#xff08;self-join&#xff09;来获取上一条记录的结束时间和下一条记录的开始时间&#xff0c;并将它们组合成一条记录。首先&#xff0c;需要为表创建一个包含记录ID和时间信息的临时表&#xff0c;然后使用自连接获取相邻记录的时间…

【面试经典150 | 数组】移除元素

文章目录 写在前面Tag题目来源题目解读解题思路方法一&#xff1a;原地操作 写在最后 写在前面 本专栏专注于分析与讲解【面试经典150】算法&#xff0c;两到三天更新一篇文章&#xff0c;欢迎催更…… 专栏内容以分析题目为主&#xff0c;并附带一些对于本题涉及到的数据结构等…

【STM32】常用存储器

常用存储器 RAM 存储器 RAM 是“Random Access Memory”的缩写&#xff0c;被译为随机存储器。所谓“随机存取”&#xff0c;指的是当存储器中的消息被读取或写入时&#xff0c;所需要的时间与这段信息所在的位置无关。而RAM可随读取其内部任意地址的数据&#xff0c;时间都是…

若依框架i18n国际化

需求背景 项目需求要做国际化&#xff0c;结果网上找了好几篇文章&#xff0c;没有一个可以一次性搞定&#xff0c;现在这里总结一下。首先&#xff0c;我们分为两部分处理&#xff0c;一个是前端页面的静态文字&#xff0c;这个由前端vue.json自行处理。第二部分就是后端的错…

Qt下SVG格式图片应用

SVG格式图片介绍 svg格式图片又称矢量图&#xff0c;该种格式的图片不同于png等格式的图片&#xff0c;采用的并不是位图的形式来组织图片&#xff0c;而是采用线条等组织图片&#xff0c;svg格式是图片的文件格式是xml&#xff0c;可以通过文件编译器打开查看svg格式内容。 …

使用Vagrant创建和管理本地Kubernetes(K8s)集群的步骤是什么

文章目录 步骤1&#xff1a;准备环境步骤2&#xff1a;创建Vagrantfile步骤3&#xff1a;启动虚拟机步骤4&#xff1a;安装Kubernetes步骤5&#xff1a;配置Kubernetes网络插件步骤6&#xff1a;将Worker节点加入集群步骤7&#xff1a;验证集群步骤8&#xff1a;部署应用步骤9&…

综合续航达1040公里:腾势计划2024年在香港上市,售价60-100 万

腾势汽车表示&#xff0c;他们计划于2024年在香港地区上市全新的D9车型。这款中大型高端新能源MPV是通过DM-i超级混动技术打造的&#xff0c;由于综合续航能力达到1040公里&#xff0c;且纯电续航最大可达190公里&#xff0c;这款车已经引起了广泛关注。据腾势销售事业部总经理…

[小尾巴 UI 组件库] 全屏响应式轮播背景图(基于 Vue 3 与 Element Plus)

文章归档于&#xff1a;https://www.yuque.com/u27599042/row3c6 组件库地址 npm&#xff1a;https://www.npmjs.com/package/xwb-ui?activeTabreadme小尾巴 UI 组件库源码 gitee&#xff1a;https://gitee.com/tongchaowei/xwb-ui小尾巴 UI 组件库测试代码 gitee&#xff1a…

在ExoPlayer中使用协程:构建强大的Android媒体播放器

在ExoPlayer中使用协程&#xff1a;构建强大的Android媒体播放器 现今的移动应用世界中&#xff0c;媒体消费是用户体验的核心部分。无论是流媒体视频、音乐播放还是处理自适应媒体格式&#xff0c;强大的媒体播放器对于提供无缝和愉悦的用户体验至关重要。而在安卓平台上&…

C高级day4循环语句

1&#xff0c;思维导图 运行结果为&#xff1a; 运行结果为&#xff1a;

CSS读书笔记

——————————————精华部分—————————————— 1、选择器 &#xff08;1&#xff09;基本选择器&#xff1a; 标签选择器 body{} 类选择器 class .class名称{} ID选择器 id #id名称{} 优先级&#xff1a;ID选择器 > 类选择器 > 标签选择器 &am…

cf 交互题

今天cf遇到了交互题&#xff0c;这个交互题的算法很很很简单&#xff0c;但是在交互上卡了&#xff0c;导致交上的代码都不算罚时。&#xff08;更伤心了。 所以&#xff0c;现在写一下交互题的做法&#xff0c;印象深刻嘛。 交互题&#xff0c;就是跟机器进行交互。你代码运…

道路积水监测-路面积水监测系统

随着城市化的不断发展&#xff0c;城市面临着越来越多的交通挑战&#xff0c;其中之一就是道路积水问题。道路积水不仅影响了交通安全&#xff0c;还会引发交通堵塞、交通事故和城市洪涝等问题。因此&#xff0c;开展道路积水监测是十分必要的。 城市排水、供水、燃气、供热、桥…

基于大规模MIMO通信系统的半盲信道估计算法matlab性能仿真

目录 1.算法运行效果图预览 2.算法运行软件版本 3.部分核心程序 4.算法理论概述 5.算法完整程序工程 1.算法运行效果图预览 2.算法运行软件版本 matlab2022a 3.部分核心程序 %EM算法收敛所需的迭代 nIter 1; Yp Y(:,1:L_polit,:); %与导频序列相对应的部分 q…

建议收藏!TCP协议面试灵魂12 问

先亮出这篇文章的思维导图: TCP 作为传输层的协议&#xff0c;是一个IT工程师素养的体现&#xff0c;也是面试中经常被问到的知识点。在此&#xff0c;我将 TCP 核心的一些问题梳理了一下&#xff0c;希望能帮到各位。 001. 能不能说一说 TCP 和 UDP 的区别&#xff1f; 首先…

如何查询成绩或工资

为什么每次查询成绩或者工资的时候都觉得麻烦又耗时呢&#xff1f;在过去&#xff0c;我们可能需要去学校或公司的相关部门&#xff0c;填写繁琐的表格&#xff0c;然后等待工作人员进行查询和处理。这不仅浪费了我们宝贵的时间&#xff0c;还可能出现查询结果不准确或者遗漏的…

芯科蓝牙BG27开发笔记4-SSV5 IDE的使用

1. 如何转移工作区的项目文件到新的文件夹&#xff0c;并且可以继续使用ssv5编辑、编译&#xff1f; 从默认的工作区将目标工程整体拷贝出来 目标文件夹&#xff1a; 进入ssv5点击导入工程&#xff0c;并选择目标文件夹 继续下一步&#xff0c;修改项目文件夹所在位置为其源码…

tcp与udp

tcp 服务端回复完SYNACK之后&#xff0c;就建立连接 1.为什么是三次&#xff0c;而不是两次&#xff1f;服务端回复完SYNACK之后&#xff0c;就建立连接 这是为了防止因为已失效的请求报文&#xff0c;突然又传到服务器引起错误 意思就是&#xff1a;假设采用两次握手建立连…

聚观早报|华为Mate 60 Pro支持面容支付;特斯拉重回底特律车展

【聚观365】9月8日消息 华为Mate 60 Pro已支持面容支付 特斯拉将重回底特律车展 iPhone在美国有1.67亿用户 韩国半导体8月份出口85.6亿美元 比亚迪元PLUS冠军版将于9月15日上市 华为Mate 60 Pro已支持面容支付 毫无预热的华为Mate 60 Pro突然在华为商城首批开售&#xf…