首页 > 其他分享 >Android注解开发APT

Android注解开发APT

时间:2024-07-16 20:28:47浏览次数:11  
标签:String typeElement APT processingEnv 注解 Android subMethod 节点

文章目录

  • 简介
  • APT注解开发工具
    • 使用场景
    • 开发流程
      • 前情提要
        • 创建aar
        • 创建jar
      • 声明注解
      • 注解编译处理
        • gradle
        • Processor
          • process方法
          • 完整举例
    • 编译结果
  • 总结

简介

相信各位Android开发对注解这个东西并不陌生,毕竟在用Arouter的时候,注解@Route是基本要使用的注解。但这只是对注解的使用。
那么在日常开发中,如何去定义注解,并通过注解在编译期间去生成一些代码,这便是本文要介绍的APT技术。

APT注解开发工具

使用场景

对于一些格式相对固定的代码,可能并不希望开发者去花费时间去编写这些固定的代码。针对这种场景,就可以考虑使用注解开发。通过声明注解,给定相对应的参数,并在编译期生成这些固定的代码。

举个简单的例子,让我们来简化一下Arouter的流程:
场景描述:一个模块内有多个类需要路由,路由表维护了名称和类的全类名用于反射(简单来说就是有个map,key是路由名称,value时全类名),那么就可以通过名称获取到对应的类。

开发流程

注解开发通常涉及两个部分(最终产物是俩jar包):

  1. 注解声明部分, annotation
  2. 注解编译处理部分,compile

前情提要

这点很重要,在这吃过亏,耽误过一段时间
library的类型有aar和jar两种
aar和安卓相关,jar和java(kotlin)相关
aar可以引用jar,但jar不能引用aar

创建aar

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-Zxo5qJQQ-1721132940679)(https://i-blog.csdnimg.cn/direct/1dfdd0a098b1463a9fd861fcab1a9e40.png#pic_center)]

创建jar

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-qifbNTkJ-1721132940681)(https://i-blog.csdnimg.cn/direct/5b22a298c98941adb49c6078a395eeca.png#pic_center)]

声明注解

注解的声明十分简单,没有什么坑,本文介绍场景都是对类的注解(其他的没研究,偷个懒)。

@Target(ElementType.TYPE)  
@Retention(RetentionPolicy.CLASS)  
public @interface MethodBridge {  
    String subMethod();  
}

MethodBridge这就是一个注解的名称,String subMethod()定义了注解内部包含的属性,这在编译期会被用到。

OK注解声明部分就这么些玩意。

注解编译处理

这是注解开发的核心部分,所有注解处理的相关操作都是在这里执行的

gradle

首先,这玩意是需要在编译期生成代码的,所以kapt插件是得有的。我现在用的是ktx版本的gradle,语法如下:

plugins {  
    kotlin("kapt")  
}

依赖呢,直接抄作业就行:

dependencies {  
    implementation(fileTree("libs"){  
        include("*.jar,*.aar")  
    })  
    implementation("com.google.auto.service:auto-service:1.0-rc6")  
    kapt("com.google.auto.service:auto-service:1.0-rc6")  
}

注意一下这个dependencies里引一下上一节定义的annotation包,因为解析的就是这里的注解。

Processor

可以将这部分理解为注解处理器
大体结构为:

@AutoService(Processor.class)  
@SupportedSourceVersion(SourceVersion.RELEASE_8)  
public class MethodBridgeCompile extends AbstractProcessor {  
  
    private String moduleName = "CommonBridgeModule";  
  
    Filer filer;  
    Messager messager;  
  
    @Override  
    public synchronized void init(ProcessingEnvironment processingEnv) {  
        super.init(processingEnv);  
        messager = processingEnv.getMessager(); // 打印对象  
        filer = processingEnv.getFiler(); // 文件操作对象  
        messager.printMessage(Diagnostic.Kind.NOTE, "moduleName:::" + processingEnv.getOptions().get("BRIDGE_MODULE_NAME"));  
        if (processingEnv.getOptions().get("BRIDGE_MODULE_NAME") != null){  
            moduleName = removeInvalidChars(processingEnv.getOptions().get("BRIDGE_MODULE_NAME"));  
        }  
    }  
  
    /**  
     * 获取注解类型  
     */  
    @Override  
    public Set<String> getSupportedAnnotationTypes() {  
        HashSet<String> types = new HashSet<>();  
        types.add(MethodBridge.class.getCanonicalName());  
        return types;  
    }  
  
    /**  
     * 声明支持的Java版本  
     */  
    @Override  
    public SourceVersion getSupportedSourceVersion() {  
        return processingEnv.getSourceVersion();  
    }
  
    @Override  
    public boolean process(Set<? extends TypeElement> annotations, RoundEnvironment roundEnv) {
		// 具体的注解处理逻辑
	}

	public String removeInvalidChars(String input) {  
	    return input.replaceAll("[^A-Za-z0-9]", "");  
	}
}

结构这块可以直接抄作业,主要逻辑都在process方法里。注意一下getSupportedAnnotationTypes里要给的是要处理的注解的全类名。

其次,外部声明的俩成员变量:

  1. filer用于创建类
  2. messager用于编译器在控制台打日志
messager.printMessage(Diagnostic.Kind.NOTE, "你的日志");	
process方法
  1. 首先需要获取到注解标识的类的所有节点
Set<? extends Element> elements = roundEnv.getElementsAnnotatedWith(ModuleBridge.class);
  1. 遍历节点,因为明确该注解只会用于类注解,因此每个元素都是类节点,可以强转为TypeElement
HashMap<String, String> map = new HashMap<>();
for (Element element : elements) {  
    TypeElement typeElement = (TypeElement) element;        // 节点一定是类节点,可以强转  
    String subMethod = typeElement.getAnnotation(MethodBridge.class).subMethod();  
    String clazzName = typeElement.getSimpleName().toString();  
    String package_name = processingEnv.getElementUtils().getPackageOf(typeElement).toString();  
    map.put(subMethod, package_name + "." + clazzName);  
}

简化的arouter的实现是想要将当前module内的所有注解标识类遍历完放到生成的类里,依次添加到路由表中。
那么路由表里每次添加到key对应了subMethod,value对应了全类名clazzName。都是通过类节点获取的。
typeElement.getAnnotation(MethodBridge.class).subMethod(); 用于获取注解中的参数
String clazzName = typeElement.getSimpleName().toString(); 用于获取类节点对应的类名
processingEnv.getElementUtils().getPackageOf(typeElement).toString();用于获取类节点所在包名。

  1. 创建文件
JavaFileObject javaFileObject = filer.createSourceFile(packageName + "." + newClass);  
writer = javaFileObject.openWriter();
StringBuffer buffer = new StringBuffer();
......
writer.write(buffer.toString());  
writer.close();

省略号部分,就是要插入的代码,你的buffer里写了啥,他就会在编译后生成啥。

完整举例

上面对一些关键代码简要介绍一下,下面完整点贴一下

public boolean process(Set<? extends TypeElement> annotations, RoundEnvironment roundEnv) {  
        /*  
         * 得到注解的Element节点  
         * 类节点:TypeElement  
         * 方法节点:ExecutableElement  
         * 成员变量节点:VariableElement  
         */        messager.printMessage(Diagnostic.Kind.NOTE, "MethodBridgeCompiler::process");  
        Set<? extends Element> elements = roundEnv.getElementsAnnotatedWith(MethodBridge.class);  
        // 初始化list,元素为被@MethodBridge标记的类节点  
//        Set<String> set = new HashSet<>();  
        HashMap<String, String> map = new HashMap<>();  
        messager.printMessage(Diagnostic.Kind.NOTE, "MethodBridgeCompiler开始遍历");  
  
        for (Element element : elements) {  
            TypeElement typeElement = (TypeElement) element;        // 节点一定是类节点,可以强转  
            String subMethod = typeElement.getAnnotation(MethodBridge.class).subMethod();  
            String clazzName = typeElement.getSimpleName().toString();  
            String package_name = processingEnv.getElementUtils().getPackageOf(typeElement).toString();  
            map.put(subMethod, package_name + "." + clazzName);  
        }  
  
        messager.printMessage(Diagnostic.Kind.NOTE, "map.size: " + map.size());  
  
        // 编译期构建类  
        if (!map.isEmpty()){  
            Writer writer = null;  
            Iterator<String> iterator = map.keySet().iterator();  
            String packageName = "包名";      // 固定包名  
            messager.printMessage(Diagnostic.Kind.NOTE, "moduleName: " + moduleName);  
            String newClass = moduleName + "Method";  
            try {  
                JavaFileObject javaFileObject = filer.createSourceFile(packageName + "." + newClass);  
                writer = javaFileObject.openWriter();  
                StringBuffer buffer = new StringBuffer();  
                ......
                writer.write(buffer.toString());  
                writer.close();  
            }catch (Exception e){  
                messager.printMessage(Diagnostic.Kind.NOTE, "exception: " + e.getMessage());  
            }  
        }  
        return false;  
    }

编译结果

集成后走一下编译,产物会在如下目录
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-plr5xA6q-1721132940682)(https://i-blog.csdnimg.cn/direct/d412cbfe6cb14414a1163ed8ced85781.png#pic_center)]

总结

基本的使用就是这样,可以试一下在buffer里随便给一些字符串看一下效果。map的逻辑也可以按照自己的逻辑选择是否添加。

标签:String,typeElement,APT,processingEnv,注解,Android,subMethod,节点
From: https://blog.csdn.net/qq_42430654/article/details/140475892

相关文章

  • bundletool工具使用(Android aab包安装)
    ......
  • 2024-07-16升级问题:调用自带软件打开文件时 android.os.FileUriExposedException
    2024-07-16升级问题:调用手机自带软件打开文件时,出现以下问题:E/AndroidRuntime:FATALEXCEPTION:mainProcess:rs.tabletcropland,PID:10997android.os.FileUriExposedException:file:///storage/emulated/0/arcgis/%E7%9F%B3%E7%8B%AE%E5%B8%82/Attachment/%E7......
  • 一个专为Android平台设计的高度可定制的日历库
    大家好,今天给大家分享一个高度可定制的日历库kizitonwose/Calendar。Calendar专为Android平台设计,支持RecyclerView和Compose框架。它提供了丰富的功能,允许开发者根据需求定制日历的外观和功能。项目介绍此库是开发Android应用时,实现日历功能的一个强大工具,特别适合那些需要......
  • android学习day2
    activity是应用程序的组件xml:描绘应用界面java:编写程序逻辑1.完整页面的创建过程:在layout目录下创建xml文件创建xml文件对应的java代码在AndroidManifest中注册页面配置 <?xmlversion="1.0"encoding="utf-8"?><manifestxmlns:android="http://schemas.android......
  • android学习day1
    1.android系统框架android大致可分为四层架构:linux内核层,系统运行库层,应用框架层和应用层1.1linux内核层为android设备的各种硬件提供底层驱动,如显示驱动,音频驱动,wifi驱动,电源管理等。1.2系统运行库层通过一些c/c++库为android系统提供了主要的特性支持,如SQLite库提供数......
  • 记一次Burp与NEW_xp_CAPTCHA工具联动爆破验证码
    首先下载NEW_xp_CAPTCHA工具地址:https://github.com/smxiazi我下载的是大佬直接发布的打包好的环境,包括对应python3.6.6与NEW_xp_CAPTCHA工具脚本下载完后直接点击运行即可本地访问http://127.0.0.1:8899/,看到这个页面,证明没问题然后就是burp导入插件jar。这里要下载xp_CA......
  • 【Android面试八股文】1. 说一说Java四大引用有哪些? 2. 软引用和弱引用的区别是什么?
    一、Java四大引用有哪些?在Java中,有四种不同类型的引用,它们在垃圾回收和对象生命周期管理方面有着不同的作用和行为。这四种引用分别是:强引用(StrongReference)软引用(SoftReference)弱引用(WeakReference)虚引用(PhantomReference)下面详细解释每种引用的特点和用途:......
  • springboot常用注解大全(超详细, 30个)
    SpringBoot注解主要用于简化配置、自动装配组件和实现声明式服务。以下是详细的介绍:1、Springboot注解核心注解1.@SpringBootApplication作用:标注一个主程序类,表明这是一个SpringBoot应用程序的入口。功能:这是一个复合注解,组合了@Configuration、@EnableAutoConfigur......
  • Java中的反射与注解结合使用
    Java中的反射与注解结合使用大家好,我是微赚淘客系统3.0的小编,是个冬天不穿秋裤,天冷也要风度的程序猿!反射与注解的基础概念在Java编程中,反射(Reflection)和注解(Annotation)是两个强大的特性,它们可以让我们在运行时获取类的信息并动态操作类的成员,同时可以通过元数据为代码添加标记......
  • android系统启动流程- ServiceManager进程启动流程
    *注:基于Android11源码ServiceManager进程是在init进程创建的,所以我们从init进程的main()开始分析://文件路径:system/core/init/main.cppintmain(intargc,char**argv){...if(!strcmp(argv[1],"second_stage")){//TODO根据条件会走到这个分支......