java APT原理及APT实战 - 一步步教你写ButterKnife

news2024/12/26 23:34:46

一、定义

Java APT 是 Java 技术设计的一个 APT 架构,

APT(Annotation Processing Tool)即注解处理器,它是一种处理注解的工具,也是javac中的一个工具,用于在编译阶段未生成class之前对源码中的注解进行扫描和处理。

APT可以用来在编译时扫描和处理注解, 它可以用来获取到注解和被注解对象的相关信息,在拿到这些信息后我们可以根据需求来自动的生成一些代码,省去了手动编写。 在Android中有如ButterKnife、Dagger、EventBus等第三方框架,都采用了APT。

常用的使用方式是这样的:

APT + 注解处理器(AbstractProcess)+ 代码处理(javaPoet)+ 处理器注册(AutoService)

这里我们了解下相关概念:

APT

是一个工具,可以用于检查源代码中的注解,并生成相应的代码。

AbstractProcess:

是 Java 中的一个抽象类,用于定义一个处理程序或者一个抽象的执行环境。在 Java 的编程过程中,经常需要在类的开头或者方法中加入注解,来标记这个方法或者类是抽象的,或者是实现了某个接口。这时候,AbstractProcess 类就可以用来定义注解处理器,用来处理这些注解。

APT框架中的核心类是AbstractProcessor,它定义了如何在代码中处理注解。当APT工具扫描文件时,它会调用实现AbstractProcessor类的一个或多个类的process()方法,以捕获和处理注解。

二、APT工作原理

Java Annotations Processing Tool(APT)是一个预处理器,可以在Java代码编译期间读取注解,并生成相关的代码。

它的工作原理如下:

  1. Java编译器会将源文件传递给APT进行处理;
  2. APT会扫描源文件中所有的注解,并找到对应的处理器;
  3. 处理器会对注解进行处理,并生成新的Java代码文件(或其他文件);
  4. 生成的Java代码文件被编译成字节码文件;
  5. 编译器将生成的字节码文件和原始Java代码文件一起打包成jar包或class文件。

APT通过Java标准类库中的javax.annotation.processing包提供注解处理的框架。注解处理器必须实现该包中的特定接口,这些接口定义了APT框架的核心功能。通过实现接口,注解处理器能够直接访问来自编译器的数据,以及用于注解处理的元数据信息。注解处理器总是运行在 Java 编译环境中。

APT的使用可以帮助简化一些重复、冗杂的代码生成工作。

三、APT实战1(运行时注解) - 一步步教你写ButterKnife

使用运行时注解的方式实战简单APT,这个比较简单,就是用反射的方式来实现,

void injectLayout(Context context) {
        // 1. 获取当前class
        Class<?> clazz = context.getClass();
        // 2. 根据class获取class上面的注解
        InjectContenttLayout annotation = clazz.getAnnotation(InjectContenttLayout.class);

        // 3. 获取注解中布局文件的id的值
        int layoutId = annotation.value();
        try {
            // 4. 获取activity中的setContentView方法
            Method method = clazz.getMethod("setContentView", int.class);
            // 5. 执行setContentView方法,传入layoutId参数
            method.invoke(context, layoutId);
        } catch (Exception e) {
        }
    }

四、APT实战2(编译时注解) - 一步步教你写ButterKnife

使用编译时注解的方式实战简单APT,手写ButterKnife框架,我们来写一下布局文件view的注入,比如我们不想写烦人的findViewById方法,直接用个注解来搞定,

本 Demo 下载

其核心思想是java的ioc(inversion of control),也叫di(dependency injection,依赖注入),是一种面向对象编程中的设计模式。下面我们开始

4.1 创建一个项目,如下图

4.2 新建自定义注解

创建一个Java Library Module名称叫 apt-annotation

在这个module中创建自定义注解 @BindView,如下图

4.3 实现APT Compiler处理注解

创建一个Java Library Module名称叫 apt-compiler-processor,并添加注解module依赖

dependencies {
    implementation project(':apt-annotation')
}

这个module的作用主要是用来处理注解,并生成java帮助类文件,拆解步骤为

1、扫描所有被注解标记的Element,获得注解标记的element

2、遍历Element,并根据不同的页面进行分类

3、按规则进行字符串拼接,用于拼接生成帮助类代码

4、使用JavaFileObject写入文件生成java代码

(输出的文件在build->generated-ap_generated_sources->debug->out->包名目录下)

如下图所示

该module在处理注解时,必须继承AbstractProcessor抽象类,入口类为process(Set<? extends TypeElement> set, RoundEnvironment roundEnv)


    /**
     * 注解处理方法,
     *
     * @param set      注解的类型集合
     * @param roundEnv 运行环境
     * @return
     */
    @Override
    public boolean process(Set<? extends TypeElement> set, RoundEnvironment roundEnv) {
        System.out.println("start process");
        if (set != null && set.size() != 0) {
            //1、 扫描所有被注解标记的Element,获得被BindView注解标记的element
            Set<? extends Element> elements = roundEnv.getElementsAnnotatedWith(BindView.class);

            categories(elements);

            for (TypeElement typeElement : mToBindMap.keySet()) {
                // 获取帮助类所有代码
                String code = generateCode(typeElement);
                // 构建要生成的帮助类的类名
                String helperClassName = typeElement.getQualifiedName() + "_ButterKnifeTest"; //

                System.out.println("start process 帮助类的类名= " + helperClassName);

                // 输出帮助类的java文件,
                // 在本例中就是MainActivity_ButterKnifeTest.java文件
                // 输出的文件在build->generated-ap_generated_sources->debug->out->包名目录下
                try {
                    System.out.println("生成帮助类 ");
                    JavaFileObject jfo = mFilerUtils.createSourceFile(helperClassName, typeElement);
                    Writer writer = jfo.openWriter();
                    writer.write(code);
                    writer.flush();
                    writer.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }

            }
            return true;
        }
        return false;
    }

当代码写完后,就需要注册APT(如上图所示)

注册一个APT需要以下步骤:

  1. 在main 目录下新建 resources 资源文件夹;
  2. 在 resources文件夹下建立 META-INF/services 目录文件夹;
  3. 在 META-INF/services 目录文件夹下创建 javax.annotation.processing.Processor 文件;
  4. 在 javax.annotation.processing.Processor 文件写入注解处理器的全称,包括包路径;

4.4 以上步骤全部完成后, 还需要对外提供API(当然也可以不拆分)

创建一个Android Library Module,名称叫apt-api,并添加依赖

dependencies {
    ...
 
    api project(':apt-annotation')
}

 实现很简单,就是通过反射去调用APT生成的帮助类的方法去实现View的自动绑定,部分代码如下:


    public void inject(Object target) {
        String className = target.getClass().getCanonicalName();
        String helperName = className + "_ButterKnifeTest";
        System.out.println("ButterKnifeTest inject" + helperName);
        try {
            IBindHelper helper = (IBindHelper) (Class.forName(helperName).getConstructor().newInstance());
            helper.inject(target);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

最后就是使用了,在app module里添加依赖

dependencies {
    ...
    annotationProcessor project(':apt-compiler-processor')
    implementation project(':apt-api')
}

4.5 使用如下:

public class MainActivity extends AppCompatActivity {

    @BindView(value = R.id.test_textview)
    public TextView testTextView;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        ButterKnifeTest.getInstance().inject(this);
        testTextView.setText("手写 butterknife demo");
    }
}

运行代码截图

 

五、痛点及优化

我们可以看到,按官方文档一步步来写apt比较繁琐,

  • 生成代码时需要字符串拼接,代码量多的时候容易出错,排查困难
  • 需要继承AbstrctProcessor并重写多个方法,写入注解容易遗漏
  • 注册APT的步骤繁琐,需手动创建文件

针对以上问题:

1、我们可以使用JavaPoet来替代拼接字符串( JavaPoet是一个用来生成Java代码的框架,对JavaPoet不了解的请自行学习)

官网地址: GitHub - square/javapoet: A Java API for generating .java source files.

2、使用Auto-Service来自动注册APT

这是谷歌官方出品的一个开源库,可以省去注册APT的步骤,只需要一行注释
先在apt-compiler模块中添加依赖

dependencies {
    ...
    
    implementation 'com.google.auto.service:auto-service:1.0-rc2'
}

然后添加注释即可,如下图所示: 

六、一些疑问

5.1 手写注解处理器时,注解处理器processor为什么要在META-INFO注册?

META-INFO相当于一个信息包,用于存放一些meta information相关的信息,用来配置应用程序、扩展程序、类加载器和服务manifest.mf文件,在编译时,java编译器回去该文件中查找实现了AbstractProcess的子类,就相当于注册。

5.2  APT(Annotation Processing Tool)如何调用AbstractProcess的呢?(注解处理器是如何被系统调用的?)

annotationProcessor 指定apt处理器。

1、创建一个类并继承自 AbstractProcessor

2、通过注解 @AutoService(Processor.class) 将创建的新类注册为 APT 处理器。

也可以手动创建,详细见下面demo说明

3、在 build.gradle 文件或项目的构建文件中指定 APT 处理器。

annotationProcessor project(':apt-compiler-processor')

4、运行项目来激活 APT 处理器并让其开始处理。

    在 Eclipse 中你可以设置 Java Compiler > Annotation Processing 选项卡中的 Enable annotation processing 复选框来启用它,

    在 Intellij IDEA 中,你需要选择菜单中的 "Build" > "Rebuild project" 来刷新生成器任务列表。完成这些步骤后,就可以开始使用 AbstractProcessor 创建自己的注解处理器。

    在安卓中,直接构建项目即可。

5.3 安卓中,APT项目会不会增加apk的体积?

不会,processor的作用是在编译器解析注解、生成文件等,只在编译器用到,是不会打包进apk的。

更高级的用法可自行阅读开源项目,向大佬学习~

Demo 下载 :https://download.csdn.net/download/fumeidonga/87767415

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

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

相关文章

Windows巧用git实现笔记自动备份

Windows巧用git实现笔记自动备份 准备git仓库配置自动上传脚本设置 Windows 自动定时任务参考文献 准备git仓库 安装git&#xff1a;https://git-scm.com/downloads&#xff1a; 注册并登录gitee&#xff0c;本地生成ssh key&#xff08;详情百度&#xff09;&#xff0c;然后…

数据处理Pandas学习笔记(一)

import pandas as pdpandas值series创建 t pd.Series([1, 2, 31, 12, 3, 4]) t0 1 1 2 2 31 3 12 4 3 5 4 dtype: int64type(t)pandas.core.series.Seriesseries指定索引 t2 pd.Series([1,23,3,2,3],indexlist(abcde)) t2a 1 b 23 c 3 d …

【Java-05】常用API、正则表达式、Collection集合

主要内容 BigInteger类BigDecimal类Arrays类包装类String类的常用方法正则表达式Collection集合 1 BigInteger类 1.1 概述 概述 : java.math.BigInteger类是一个引用数据类型 , 可以用于计算一些大的整数 , 当超出基本数据类型数据范围的整数运算时就可以使用BigInteger了。…

类别无关的姿态估计ECCV2022

现有的2D姿态估计工作主要集中在某一类别&#xff0c;例如人类、动物和车辆。然而&#xff0c;有很多应用场景需要检测unseen对象类的姿态&#xff08;或关键点&#xff09;。因此作者提出CAPE任务&#xff08;Category-Agnostic Pose Estimation&#xff09;&#xff0c;该任务…

Sketch哪个版本好用?

使用最新版本的 Sketch 是很有意义的。一方面&#xff0c;最新版本通常会有新的功能和改进&#xff0c;使设计师更方便地完成工作。另一方面&#xff0c;使用最新版本还可以避免出现因版本不兼容而无法打开源文件的问题。此外&#xff0c;最新版本通常会更稳定&#xff0c;因此…

Linux command(sar)

说明 sar命令是一个系统性能监测工具&#xff0c;用于收集、报告和分析系统的各种资源使用情况。以下是sar命令的基本用法&#xff1a; sar [选项] [时间间隔] [次数] 选项&#xff1a;可用的选项包括-a&#xff08;显示所有资源使用情况&#xff09;、-b&#xff08;显示I/…

图片修复增强调研

Real-ESRGAN 工程地址&#xff1a;https://github.com/xinntao/Real-ESRGAN 效果&#xff1a; 人脸增强部分&#xff0c;调用的GFPGAN. GFPGAN 工程地址&#xff1a;https://github.com/TencentARC/GFPGAN 论文效果&#xff1a; BasicSR-ESRGAN&#xff1a; 项目地址&a…

[Qt编程之Widgets模块] -001: QButtonGroup抽象容器

1.QButtonGroup简介 QButtonGroup提供了一个抽象容器&#xff0c;可以将按钮小部件放入其中。它不提供此容器的可视化表示&#xff0c;而是管理组中每个按钮的状态。 互斥按钮组&#xff0c;将关闭除已单击的按钮外的所有可选中&#xff08;可切换&#xff09;按钮。默认情况下…

免费使用GPT-4.0?【AI聊天 | GPT4教学】 —— 微软 New Bing GPT4 申请与使用保姆级教程

目录 认识 New Bing 2. 注册并登录 Microsoft 账号 3. 如何免科学上网使用 New Bing&#xff1f; 4. 加入 WaitList 候补名单 5. 使用 New Bing&#xff01; 6. 使用 Skype 免科学上网访问 New Bing&#xff01; 7. 在 Chrome 浏览器中使用 New Bing&#xff01; 8. 总…

如何利用splice()和slice()方法操作数组

如何利用splice&#xff08;&#xff09;和slice&#xff08;&#xff09;方法操作数组 前言splice()是什么&#xff0c;有什么用&#xff1f;怎么用&#xff1f;slice()是什么&#xff0c;有什么用&#xff1f;怎么用&#xff1f;splice和slice方法的区别小结 前言 splice&am…

如今的Android就业率惨不忍睹~

3月底公司大裁员&#xff0c;投了一个月简历&#xff0c;一天投个几十份简历&#xff0c;而收到面试通知的就那么三四家&#xff0c;要么就是薪水给得很低不想去&#xff0c;要么就是高薪水的Offer拿不下&#xff0c;而自己中意公司的却没有给出回应,唉……真难啊&#xff01;&…

4.2 线性表顺序表

目录 目录结构 线性表 线性表的特征&#xff1a; 顺序表存储结构的表示 顺序表存储结构的特点 顺序存储结构的表示 线性表的基本运算 基本运算的相关算法 线性表的基本运算 线性表 目录结构 线性表 线性表是包含若干数据元素的一个线性序列 记为&#xff1a; L(a0, …

Istio virtual service 故障注入之延时(fixedDelay)、中断(abort)

Istio 故障注入 Istio 故障注入与其他在网络层引入错误&#xff08;例如延迟数据包或者直接杀死 Pod&#xff09;的机制不同&#xff0c;Istio 允许在应用程序层注入故障。这使得可以注入更多相关的故障&#xff0c;比如 HTTP 错误代码等。 Istio 可以注入两种类型的故障&…

iLO初识

文章目录 什么是iLO&#xff1f;为什么需要iLO&#xff1f;iLO界面介绍风扇状态温度状态存储控制器的电池组件cpu状态内存状态网络设备列表存储状态固件状态 什么是iLO&#xff1f; iLO&#xff08;Integrated Lights-Out&#xff09;是一种远程管理技术&#xff0c;用于管理和…

第7章链接:7.1 编译器驱动程序

示例程序由两个源文件组成&#xff0c;main.c 和 swap.c。 main函数初始化一个两元素的整数数组&#xff0c;然后调用swap函数来交换这一对数。 main.c void swap();int buf[2] {1, 2};int main() {swap();return 0; }swap.c extern int buf[];int *bufp0 &buf[0]; i…

Android系统日志3-trace

1、使用 主要追踪耗时&#xff0c;如上图所示&#xff0c;记录的是traceBegin和traceEnd之间的耗时&#xff0c;这些日志信息可以通过systrace查看 2、获取trace日志 https://developer.android.com/topic/performance/tracing/command-line?hlzh-cn https://developer.andr…

Linux之shell基础(六)

1、shell 简介 shell 介于内核与用户之间&#xff0c;负责命令的编辑&#xff0c;是人机交互的一个桥梁 shell 的种类 [rootyhm ~]# cat /etc/shells /bin/sh /bin/bash /usr/bin/sh /usr/bin/bash /bin/ksh /bin/rksh /usr/bin/ksh /usr/bin/rksh种类说明/bin/shbash的一个…

学习HCIP的day.05

目录 十、ospf扩展配置&#xff1a; 1、认证 2、沉默接口 3、加快收敛 4、缺省路由 查看OSPF路由&#xff1b;显示本地发出和学习到的OSPF路由条目 <r10>display ospf routing 十、ospf扩展配置&#xff1a; 1、认证 直连的邻居或邻接之间&#xff0c;配置身…

【并发编程系列 — 线程基础】

Java并发是Java语言中非常重要的一个话题&#xff0c;也是面试和实际应用中最常接触到的。作为JAVA工程师&#xff0c;掌握并发编程知识是非常重要的。在多核CPU的环境下&#xff0c;使用多线程并发执行任务可以显著提高程序的执行效率&#xff0c;从而提高系统的性能。 目录 …

6.事件系统

事件 事件的传递 新建Widget项目myevent&#xff0c;类名WIdget不变 新建C类如下&#xff1a; 粘贴代码 1myevent运行结果&#xff1a; 输入任意字符时&#xff0c;qDebug显示如下&#xff0c;但框内无法显示输入的字符 //mylineedit.cpp中void MyLineEdit::keyPressEven…