【Java 基础】Java SPI 二 之 Java APT原理及APT实战 - 一步步教你写ButterKnife

简介: Java APT 是 Java 技术设计的一个 APT 架构,APT(Annotation Processing Tool)即注解处理器,它是一种处理注解的工具,也是javac中的一个工具,用于在编译阶段未生成class之前对源码中的注解进行扫描和处理。APT可以用来在编译时扫描和处理注解, 它可以用来获取到注解和被注解对象的相关信息,在拿到这些信息后我们可以根据需求来自动的生成一些代码,省去了手动编写。 在Android中有如ButterKnife、Dagger、EventBus等第三方框架,都采用了APT。

一、定义

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) {
            }
        }

    image.gif

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

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

    本 Demo 下载

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

    4.1 创建一个项目,如下图

    image.gif编辑

    4.2 新建自定义注解

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

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

    image.gif编辑

    4.3 实现APT Compiler处理注解

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

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

    image.gif

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

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

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

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

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

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

    如下图所示

    image.gif编辑

    该module在处理注解时,必须继承AbstractProcessor抽象类,入口类为process(Set 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;
        }

    image.gif

    当代码写完后,就需要注册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')
      }

      image.gif

      image.gif编辑

      实现很简单,就是通过反射去调用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();
              }
          }

      image.gif

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

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

      image.gif

      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");
          }
      }

      image.gif

      运行代码截图

      image.gif编辑

      五、痛点及优化

      我们可以看到,按官方文档一步步来写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'
        }

        image.gif

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

        image.gif编辑

        六、一些疑问

        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')
        image.gif

        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


        相关文章
        |
        11天前
        |
        监控 Java API
        探索Java NIO:究竟在哪些领域能大显身手?揭秘原理、应用场景与官方示例代码
        Java NIO(New IO)自Java SE 1.4引入,提供比传统IO更高效、灵活的操作,支持非阻塞IO和选择器特性,适用于高并发、高吞吐量场景。NIO的核心概念包括通道(Channel)、缓冲区(Buffer)和选择器(Selector),能实现多路复用和异步操作。其应用场景涵盖网络通信、文件操作、进程间通信及数据库操作等。NIO的优势在于提高并发性和性能,简化编程;但学习成本较高,且与传统IO存在不兼容性。尽管如此,NIO在构建高性能框架如Netty、Mina和Jetty中仍广泛应用。
        26 3
        |
        11天前
        |
        安全 算法 Java
        Java CAS原理和应用场景大揭秘:你掌握了吗?
        CAS(Compare and Swap)是一种乐观锁机制,通过硬件指令实现原子操作,确保多线程环境下对共享变量的安全访问。它避免了传统互斥锁的性能开销和线程阻塞问题。CAS操作包含三个步骤:获取期望值、比较当前值与期望值是否相等、若相等则更新为新值。CAS广泛应用于高并发场景,如数据库事务、分布式锁、无锁数据结构等,但需注意ABA问题。Java中常用`java.util.concurrent.atomic`包下的类支持CAS操作。
        42 2
        |
        14天前
        |
        Java
        Java基础却常被忽略:全面讲解this的实战技巧!
        本次分享来自于一道Java基础的面试试题,对this的各种妙用进行了深度讲解,并分析了一些关于this的常见面试陷阱,主要包括以下几方面内容: 1.什么是this 2.this的场景化使用案例 3.关于this的误区 4.总结与练习
        |
        2月前
        |
        存储 算法 Java
        大厂面试高频:什么是自旋锁?Java 实现自旋锁的原理?
        本文详解自旋锁的概念、优缺点、使用场景及Java实现。关注【mikechen的互联网架构】,10年+BAT架构经验倾囊相授。
        大厂面试高频:什么是自旋锁?Java 实现自旋锁的原理?
        |
        2月前
        |
        Java
        Java之CountDownLatch原理浅析
        本文介绍了Java并发工具类`CountDownLatch`的使用方法、原理及其与`Thread.join()`的区别。`CountDownLatch`通过构造函数接收一个整数参数作为计数器,调用`countDown`方法减少计数,`await`方法会阻塞当前线程,直到计数为零。文章还详细解析了其内部机制,包括初始化、`countDown`和`await`方法的工作原理,并给出了一个游戏加载场景的示例代码。
        Java之CountDownLatch原理浅析
        |
        2月前
        |
        Java 索引 容器
        Java ArrayList扩容的原理
        Java 的 `ArrayList` 是基于数组实现的动态集合。初始时,`ArrayList` 底层创建一个空数组 `elementData`,并设置 `size` 为 0。当首次添加元素时,会调用 `grow` 方法将数组扩容至默认容量 10。之后每次添加元素时,如果当前数组已满,则会再次调用 `grow` 方法进行扩容。扩容规则为:首次扩容至 10,后续扩容至原数组长度的 1.5 倍或根据实际需求扩容。例如,当需要一次性添加 100 个元素时,会直接扩容至 110 而不是 15。
        Java ArrayList扩容的原理
        |
        1月前
        |
        Java 程序员
        Java基础却常被忽略:全面讲解this的实战技巧!
        小米,29岁程序员,分享Java中`this`关键字的用法。`this`代表当前对象引用,用于区分成员变量与局部变量、构造方法间调用、支持链式调用及作为参数传递。文章还探讨了`this`在静态方法和匿名内部类中的使用误区,并提供了练习题。
        32 1
        |
        2月前
        |
        安全 Java 开发者
        Java 多线程并发控制:深入理解与实战应用
        《Java多线程并发控制:深入理解与实战应用》一书详细解析了Java多线程编程的核心概念、并发控制技术及其实战技巧,适合Java开发者深入学习和实践参考。
        67 6
        |
        2月前
        |
        存储 安全 Java
        Java多线程编程中的并发容器:深入解析与实战应用####
        在本文中,我们将探讨Java多线程编程中的一个核心话题——并发容器。不同于传统单一线程环境下的数据结构,并发容器专为多线程场景设计,确保数据访问的线程安全性和高效性。我们将从基础概念出发,逐步深入到`java.util.concurrent`包下的核心并发容器实现,如`ConcurrentHashMap`、`CopyOnWriteArrayList`以及`BlockingQueue`等,通过实例代码演示其使用方法,并分析它们背后的设计原理与适用场景。无论你是Java并发编程的初学者还是希望深化理解的开发者,本文都将为你提供有价值的见解与实践指导。 --- ####
        |
        2月前
        |
        Java 大数据 API
        14天Java基础学习——第1天:Java入门和环境搭建
        本文介绍了Java的基础知识,包括Java的简介、历史和应用领域。详细讲解了如何安装JDK并配置环境变量,以及如何使用IntelliJ IDEA创建和运行Java项目。通过示例代码“HelloWorld.java”,展示了从编写到运行的全过程。适合初学者快速入门Java编程。