转载请标明出处:https:////www.cnblogs.com/tangZH/p/12343786.html

                            http://77blogs.com/?p=199

APT 是Annotation Processing Tool 的简称。

它是注解处理器,在处理Annotation时可以根据源文件中的Annotation生成额外的源文件和其它的文件(文件具体内容由Annotation处理器的编写者决定),APT还会编译生成源文件和原来的源文件,将它们一起生成class文件。
简言之:APT可以根据注解,在编译时生成代码。

事实上它是javac的一个工具,命令行运行javac后便可以看到:

接下来我们就来实现一个apt的实例,类似于ButterKnife中@BindView注解,基本步骤如下:

1、定义要被处理的注解。

2、定义注解处理器(生成具体的类)。

3、调用处理器生成的代码

对应的,我们在工程中需要有这几个模块:

1、app。测试我们的功能

2、apt-annotation。一个Java library module,放置我们自定义注解

3、apt-processor。一个Java library module,注解处理器模块

4、apt-sdk。一个Android library module,通过反射调用apt-processor模块生成的方法,实现view的绑定。

工程目录如下:

1、在apt-annotation中自定义注解:

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target; @Retention(RetentionPolicy.CLASS)
@Target(ElementType.FIELD)
public @interface BindView {
int value();
}

2、apt-processor中引入依赖,它需要依赖apt-annotation,同时还需要依赖auto-service第三方库,后面创建注解处理器的时候需要用到。

apt-processor/build.gradle文件中:

implementation project(':apt-annotation')
implementation 'com.google.auto.service:auto-service:1.0-rc2'

3、在pat-processor中创建注解处理器:

处理器需要继承AbstractProcessor,注意该module是 java module,如果创建的是android module的话那么就会找不到AbstractProcessor

@AutoService(Processor.class)
@SuppressWarnings("unused")
public class BindViewProcessor extends AbstractProcessor {
private Elements mElementUtils;
private Map<String, ClassCreatorFactory> mClassCreatorFactoryMap = new HashMap<>(); @Override
public synchronized void init(ProcessingEnvironment processingEnvironment) {
super.init(processingEnvironment);
//拿到工具类
mElementUtils = processingEnvironment.getElementUtils();
} @Override
public Set<String> getSupportedAnnotationTypes() {
//这个注解处理器是给哪个注解用的
HashSet<String> supportType = new LinkedHashSet<>();
supportType.add(BindView.class.getCanonicalName());
return supportType;
} @Override
public SourceVersion getSupportedSourceVersion() {
//返回java版本
return SourceVersion.latestSupported();
} @Override
public boolean process(Set<? extends TypeElement> set, RoundEnvironment roundEnvironment) {
mClassCreatorFactoryMap.clear();
//得到所有包含该注解的element集合
Set<? extends Element> elements = roundEnvironment.getElementsAnnotatedWith(BindView.class); for (Element element : elements) {
//转换为VariableElement,VariableElement为element的子类
VariableElement variableElement = (VariableElement) element;
//可以获取类的信息的element,也是element的子类
TypeElement classElement = (TypeElement) variableElement.getEnclosingElement();
//获取包名加类名
String fullClassName = classElement.getQualifiedName().toString(); //保存到集合中
ClassCreatorFactory factory = mClassCreatorFactoryMap.get(fullClassName);
if (factory == null) {
factory = new ClassCreatorFactory(mElementUtils, classElement);
mClassCreatorFactoryMap.put(fullClassName, factory);
}
BindView bindViewAnnotation = variableElement.getAnnotation(BindView.class);
int id = bindViewAnnotation.value();
factory.putElement(id, variableElement);
}
//开始创建java类
for (String key : mClassCreatorFactoryMap.keySet()) {
ClassCreatorFactory factory = mClassCreatorFactoryMap.get(key);
try {
JavaFileObject fileObject = processingEnv.getFiler().createSourceFile(
factory.getClassFullName(), factory.getTypeElement());
Writer writer = fileObject.openWriter();
//写入java代码
writer.write(factory.generateJavaCode());
writer.flush();
writer.close();
} catch (IOException e) {
e.printStackTrace();
}
}
return true;
}
}

需要注意的是代码中不能有中文,否则编译不通过,我这里为了方便注释解释加上了中文。

ClassCreatorFactory的代码如下,这个类负责提供需要写入新的类的代码:

public class ClassCreatorFactory {
private String mBindClassName;
private String mPackageName;
private TypeElement mTypeElement;
private Map<Integer, VariableElement> mVariableElementMap = new HashMap<>(); ClassCreatorFactory(Elements elementUtils, TypeElement classElement) {
this.mTypeElement = classElement;
//PackageElement是element的子类,可以拿到包信息
PackageElement packageElement = elementUtils.getPackageOf(mTypeElement);
String packageName = packageElement.getQualifiedName().toString();
String className = mTypeElement.getSimpleName().toString();
this.mPackageName = packageName;
//生成的类的名称
this.mBindClassName = className + "_ViewBinding";
} public void putElement(int id, VariableElement element) {
mVariableElementMap.put(id, element);
} public String generateJavaCode() {
StringBuilder stringBuilder = new StringBuilder();
stringBuilder.append("/**\n" + " * Auto Created by apt\n" + "*/\n");
stringBuilder.append("package ").append(mPackageName).append(";\n");
stringBuilder.append('\n');
stringBuilder.append("public class ").append(mBindClassName);
stringBuilder.append(" {\n");
generateBindViewMethods(stringBuilder); stringBuilder.append('\n');
stringBuilder.append("}\n");
return stringBuilder.toString();
} private void generateBindViewMethods(StringBuilder stringBuilder) {
stringBuilder.append("\tpublic void bindView(");
stringBuilder.append(mTypeElement.getQualifiedName());
stringBuilder.append(" owner) {\n");
for (int id : mVariableElementMap.keySet()) {
VariableElement variableElement = mVariableElementMap.get(id);
String viewName = variableElement.getSimpleName().toString();
String viewType = variableElement.asType().toString();
stringBuilder.append("\t\towner.");
stringBuilder.append(viewName);
stringBuilder.append(" = ");
stringBuilder.append("(");
stringBuilder.append(viewType);
stringBuilder.append(")(((android.app.Activity)owner).findViewById( ");
stringBuilder.append(id);
stringBuilder.append("));\n");
}
stringBuilder.append(" }\n");
} public String getClassFullName() {
return mPackageName + "." + mBindClassName;
} public TypeElement getTypeElement() {
return mTypeElement;
}
}

先不谈apt-sdk模块,我们先来看看生成的代码是怎么样的。

在app的gradle中引入:

implementation project(':apt-annotation')
annotationProcessor project(':apt-processor')

特别要注意的是apt-processor模块的依赖引进要用 annotationProcessor,否则编译报错

两个activity中:

public class MainActivity extends AppCompatActivity {
@BindView(R.id.tv)
TextView textView; @BindView(R.id.tv_1)
TextView tv; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
}
}
public class Main2Activity extends AppCompatActivity {

    @BindView(R.id.tv_2)
TextView textView; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main2);
}
}

rebuild一下便可以看到在这个目录下有我们生成的文件了。

gradle高版本出现编译后没出现文件的问题,无奈只好降低版本,我使用的版本是gradle  3.1.4 +  gralde_wrap  gradle-4.4-all.zip

点进入其中一个可以看到是这样的代码:

/**
* Auto Created by apt
*/
package com.example.aptsample; public class MainActivity_ViewBinding {
public void bindView(com.example.aptsample.MainActivity owner) {
owner.tv = (android.widget.TextView)(((android.app.Activity)owner).findViewById( 2131165360));
owner.textView = (android.widget.TextView)(((android.app.Activity)owner).findViewById( 2131165359));
} }

所以我们只要调用bindView就能够找到该view了,这也是apt-sdk要做的事情。

4、在apt-sdk中创建类,反射调用生成的类中的方法

public class DataApi {
public static void bindView(Activity activity) {
Class clazz = activity.getClass();
try {
Class<?> bindViewClass = Class.forName(clazz.getName() + "_ViewBinding");
Method method = bindViewClass.getMethod("bindView", activity.getClass());
method.invoke(bindViewClass.newInstance(),activity);
} catch (ClassNotFoundException e) {
e.printStackTrace();
} catch (NoSuchMethodException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (InstantiationException e) {
e.printStackTrace();
} catch (InvocationTargetException e) {
e.printStackTrace();
}
}
}

5、app的gradle中引入apt-sdk,然后代码调用DataApi的方法

implementation project(':apt-annotation')
annotationProcessor project(':apt-processor')
implementation project(':apt-sdk')

app的MainActivity中实现

public class MainActivity extends AppCompatActivity {
@BindView(R.id.tv)
TextView textView; @BindView(R.id.tv_1)
TextView tv; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
DataApi.bindView(this);
tv.setText("a");
}
}

这样就大功告成了

精美原文:http://77blogs.com/?p=199

源码地址:https://github.com/TZHANHONG/AptAutoBindView

android的APT技术的更多相关文章

  1. Android的NDK技术

    Android的NDK技术

  2. 本人讲课时录制的Android应用开发技术教学视频

    网盘地址:http://yun.baidu.com/pcloud/album/info?query_uk=1963923831&album_id=3523786484935252365 本人讲 ...

  3. 【转】Android 防破解技术简介

    http://www.cnblogs.com/likeandroid/p/4888808.html Android 防破解技术简介 这几年随着互联网的不断发展,Android App 也越来越多!但是 ...

  4. Android推送技术研究

    前言 最近研究Android推送的实现, 研究了两天一夜, 有了一点收获, 写下来既为了分享, 也为了吐槽. 需要说明的是有些东西偏底层硬件和通信行业, 我对这些一窍不通, 只能说说自己的理解. 为什 ...

  5. 全面了解Android热修复技术

    WeTest 导读 本文探讨了Android热修复技术的发展脉络,现状及其未来. 热修复技术概述 热修复技术在近年来飞速发展,尤其是在InstantRun方案推出之后,各种热修复技术竞相涌现.国内大部 ...

  6. Android热修复技术原理详解(最新最全版本)

    本文框架 什么是热修复? 热修复框架分类 技术原理及特点 Tinker框架解析 各框架对比图 总结   通过阅读本文,你会对热修复技术有更深的认知,本文会列出各类框架的优缺点以及技术原理,文章末尾简单 ...

  7. 聊聊真实的 Android TV 开发技术栈

    智能电视越来越普及了,华为说四月发布智能电视跳票了,一加也说今后要布局智能电视,在智能电视方向,小米已经算是先驱了.但是还有不少开发把智能电视简单的理解成手机屏幕的放大,其实这两者并不一样. 一.序 ...

  8. Android 插件化技术窥探

    在Android 插件化技术中(宿主app和插件app设置相同的sharedUserId),动态加载apk有两种方式: 一种是将资源主题包的apk安装到手机上再读取apk内的资源,这种方式的原理是将宿 ...

  9. Android 防破解技术简介

    Android 防破解技术简介 这几年随着互联网的不断发展,Android App 也越来越多!但是随之而来的问题也越来越多,这其中比较令人头疼的问题就是:有些不法分子利用反编译技术破解 App,修改 ...

随机推荐

  1. SpringBoot_Web开发_定制错误数据

    SpringBoot默认的错误处理机制 默认效果: ​ 1).浏览器,返回一个默认的错误页面 2).如果是其他客户端,默认响应一个json数据 原理: ​ 可以参照ErrorMvcAutoConfig ...

  2. Visual studio 2015 与 mysql 连接

    Visual Studio 2015 Community连接到MySQL,步骤很简单,但刚弄的时候一脸. 这个学期开了一门课程,老师教的是visual studio 2010来开发.net的,但是我自 ...

  3. 一次完整的OCR实践记录

    一.任务介绍 这次的任务是对两百余张图片里面特定的编号进行识别,涉及保密的原因,这里就不能粘贴出具体的图片了,下面粘贴出一张类似需要识别的图片. 假如说我的数据源如上图所示,那么我需要做的工作就是将上 ...

  4. Shell之用户与权限

    用户与组 早期Linux系统设计为了能够实现多用户.多进程高效的利用服务器资源,在此种情况下,为了能够保证用户与用户之间的文件不被随意的访问及修改.删除等操作,用户.组的管理能在某种程序上实现管理用户 ...

  5. 单点登陆(SSO)

    一.背景 在企业发展初期,企业使用的系统很少,通常一个或者两个,每个系统都有自己的登录模块,运营人员每天用自己的账号登录,很方便.但随着企业的发展,用到的系统随之增多,运营人员在操作不同的系统时,需要 ...

  6. c++中多文件的组织

    参考书目:visual c++ 入门经典 第七版 Ivor Horton著 第八章 根据书中例子学习使用类的多文件项目. 首先要将类CBox定义成一个连贯的整体,在CBox.H文件中写入相关的类定义, ...

  7. 浏览器无法进入GitHub(已解决)

    时间:2020/1/22 今天突然chrome登不上GitHub,一直出现响应时间过长的问题,如下: 开始还以为是GitHub服务器出问题了(虽然概率很小.....),但这种情况一直持续了几个小时,我 ...

  8. 踩坑:windows系统下,nodejs版本管理器无法使用n来管理

    错误 :在windows系统下,需要npm 一个n来管理nodejs的版本,但是使用npm install -g n命令之后报错 原因 : n 不支持 windows系统  只支持mac系统.

  9. vuex源码简析

    前言 基于 vuex 3.12 按如下流程进行分析: Vue.use(Vuex); const store = new Vuex.Store({ actions, getters, state, mu ...

  10. ROS机器人话题之自定义消息

    ROS提供了丰富的内建消息,std_msgs包定义了一些基本的类型. 具体例子 首先定义一个消息类型的文件叫做Complex 例Complex.msg float32 real float32 ima ...