本文的目标是用最简代码实现 ButterKnife 的核心功能。示例代码在github上: ButterKnifeProcedure
Pluggable Annotation Processing
注解处理器
Java5 中叫APT(Annotation Processing Tool),在Java6开始,规范化为 Pluggable Annotation Processing。
第一步(收集信息)
找到所有被注解的属性或者方法,将所有的信息收集到对应的“类数据集”中。
第二步(生成源文件)
根据每一个“类数据集”,生成对应的java源文件。由于这些文件并不是在运行时生成的,因此也无需动态编译,注解处理器运行完成之后,
编译器会处理所有的编译流程。
第三步(动态注入)
运行时动态注入,即用户常规调用的 ButterKnife.bind(activity)
这一步为了避免蹩脚的调用,使用了运行时反射,但是作者对每一个类进行了缓存,因此,不会对执行效率产生多大影响。
注
在最新的 ButterKnife 源码(2015.06.08)中,ButterKnife已经重构了部分方法:
ButterKnife#inject -> ButterKnife#bind
@InjectView -> @FindView
等等,具体变化可以去看官方文档,本文档后续代码使用最新版本代码演示。
极简实现演示
演示代码说明
- 示例代码由 ButterKnife 简化而来,部分定义和实现有删改,只能绑定 Activity 中的 View 字段
- 为了避免引入Android平台,但是又需要更直观,所以mock了android的两个类,Activity 和 View
- 为了避免使用 Pluggable Annotation Processing 过程中的jar包要求,以及波及不必要的java文件,请使用命令行运行演示,直接运行 ./run.sh 即可查看结果
- 保证 CLASSPATH 中含有tools.jar
第一步(收集信息)
- 在每一个类中找到所有被 FindView 注解的字段
- 每一个需要绑定的字段信息都保存为一个 FieldViewBinding 对象,比如:
1 |
|
得到:
1 | new FieldViewBinding(vView1, android.view.View, 100) |
- 将字段分类,获取每一个类的“类数据集”BindingClass,比如, MainActivity 对应的 “类数据集” 如下:
1 | MainActivity: |
第二步(生成 Bind 工具类源文件)
为了便于在反射时容易实例化生成的类,每一个生成的类都实现了一个 ActivityBinder
1 | package sample; |
第三步(动态注入)
我们在 MainActivity 中调用 ButterKnife#bind,第一件事就是找到对应生成的 Bind 工具类,这里遵循命名规则(在对应类后增加 $$ViewBinder 后缀),直接使用动态加载并实例化:
1 | Class<?> activityBindingClass = Class.forName(targetClass.getName() + ButterKnifeProcessor.SUFFIX); |
获得相应的 ActivityBinder 之后,使用 ActivityBinder#bind 进行绑定,与手动调用 findViewById 效果相同
运行
运行:
ButterKnifeProcedure/src$ ./run.sh
结果:
mainActivity.vView1.id = 100
mainActivity.vView2.id = 200