安卓to鸿蒙系列:ButterKnife(一) 原创 精华

没用的喵叔
发布于 2021-4-30 20:05
浏览
0收藏

本文是关于ButterKnife的移植的第一篇:

先介绍基础知识,理解apt是什么,最终输出一个乞丐版BufferKnife注入工具。

这是读懂BufferKnife源码的基础。

第二篇占坑:具体讲解BufferKnife的移植。


基础知识

apt技术的实际应用:

  • Android Databinding绑定view

  • ButterKnife绑定view

  • Dagger2注入变量

  • ARouter生成路由表

真的好多,很重要。apt就是生产力工具!

什么是apt?

通俗讲:apt就是javac对外开放的一个插件,使javac在编译期间获取注解(Annotation),并做出相应的处理(多数都是生成一些java代码)。

安卓to鸿蒙系列:ButterKnife(一)-鸿蒙开发者社区

从上图可以看出apt处理的是 java源文件 ,在编译期介入。

与之对比的是asm之类的工具,处理的是字节码文件,在编译后期介入。

apt与javac的约定

apt与javac约定在META-INF/services/javax.annotation.processing.Processor文件中注册apt插件。这样apt就参与到javac的编译过程中了。

写一个乞丐版BufferKnife

不想看文字,直接看源码,点击 https://gitee.com/andych008/aptDemo

初始代码fork自 https://github.com/LiMubai2017/aptDemo ,先对作者表示感谢。


乞丐版BufferKnife作为一款view注入工具,主要干了3件事,

  1. 解析注解
  2. 处理注解(生成模板类文件)
  3. 通过模板类注入view对象

第0步:准备工作

先定义注解BindView,被@BindView标记的变量会被注入。

一般注解都定义在一个单独的module(如取名apt-annotation),因为它会被apt-compiler和apt-api都依赖,属于公共代码。

apt-compiler是apt的主要代码所在,完成注解的解析、模板文件的生成。

apt-api则是对外的工具类,供用户使用,完成注入操作。

app是demo,其中定义了

@BindView(value = ResourceTable.Id_text_helloworld)
public Text testTextView;

第一步:解析注解

在apt-compiler中定义类BindViewProcessor继承javax.annotation.processing.AbstractProcessor,实现其中的getSupportedAnnotationTypes()该方法注册要解析的注解。

第二步:处理注解(生成模板文件)

BindViewProcessor中实现process()方法,处理注解。

先理解javax.lang.model.element.Elementjavax.lang.model.type.TypeMirror,参考这里 有详细的解释。

简单讲:

Element是描述java语言元素的类,比如包、类、变量、参数等。

TypeMirror是描述Element类型的类,比如各种基本类型、数组、类等。

很绕,只有多用才能真正理解。比如:demo中testTextView就是VariableElement元素类型

TypeElement enclosingElement = (TypeElement) variableElement.getEnclosingElement();//获取代表MainAbility的TypeElement

String field = variableElement.getSimpleName().toString();//testTextView
TypeMirror typeMirror = variableElement.asType();//ohos.agp.components.Text

通过log()方法,可以使用Messager打日志,验证我们的理解。

log(String.format("element : (%s) %s ", element.getKind(), element));
log(String.format("bind : (%s) %s <--> id = %d", typeMirror, field, id));
输出日志:
注: element : (FIELD) testTextView 
注: bind : (ohos.agp.components.Text) testTextView <--> id = 16777222

generateCodeByPoet()方法中,使用javapoet生成模板代码MainAbility$$Autobind.java(文件路径app/build/generated/source/annotation/debug/com/example/apt_demo/MainAbility$$Autobind.java

关于javapoet的使用,直接看官方文档吧:https://github.com/square/javapoet

解释一下下面这段代码,让大家对javapoet有一直观的认识

MethodSpec.Builder injectMethod = MethodSpec.methodBuilder("inject")//生成一个方法,方法名是inject
        .addAnnotation(Override.class)//给方法加上"Override.class"注解
        .addModifiers(Modifier.PUBLIC)//给方法加上访问控制符
        .addParameter(Object.class, "target")//给方法加上参数
        .addStatement("$T substitute = ($T)target", className, className);//在方法体内定义一条语名

上面的代码生成下面的代码(我用java代码生成java代码,这就是javapoet干的事情):

@Override
public void inject(Object target) {
  MainAbility substitute = (MainAbility)target;
}

看完上面这一坨,你如果觉得难。请用JavaWriter生成java文件。你就会觉得javapoet真香。

第三步:通过模板类注入view对象

在apt-api中,我们定义一个AutoBind.java类封装对模板类MainAbility$$Autobind.java的操作。

按照模板类的命名规则xxx$$Autobind,通过反射实例化出MainAbility$$Autobind.java,调用 其中的inject方法,完成view的注入。

总结

apt只是一个工具,在这套工具框架下,怎么处理注解才是难点。

BufferKnife和我们的“乞丐版BufferKnife”本质上没有区别。除了注入view,还支持事件绑定、增量编译。

©著作权归作者所有,如需转载,请注明出处,否则将追究法律责任
标签
已于2021-5-6 22:56:46修改
3
收藏
回复
举报
3条回复
按时间正序
/
按时间倒序
没用的喵叔
没用的喵叔

下一篇基于https://gitee.com/openharmony-tpc/butterknife 分析bufferknife的移植。

1
回复
2021-4-30 20:07:58
张荣超_九丘教育
张荣超_九丘教育

👍👍👍

回复
2021-5-1 10:39:20
我来抓人啦
我来抓人啦

感谢贡献哈。加油

回复
2021-6-2 16:59:22
回复
    相关推荐