代码自动生成
- List item使用代码自动生成,一是为了提高编码的效率,二十避免在运行期大量使用反射,通过在编译期利用反射生成辅助类和方法以供运行时使用
注解处理器步骤
- 在java编译器中构建
- 编译器开始执行未执行过的注解处理器
- 玄幻处理注解元素,找到被该住解所修饰的类,方法或者属性
- 生成对应的类,并写入文件
- 判断是否所有的注解处理器都已执行完毕,如果没有,继续下一个注解处理器的执行(回到步骤1)
注解处理器的实现
-
主要需要两个库的AutoService和Javapoet,对所有的注解进行处理并生成以_BindView结尾的辅助类辅助类中是注解的实现(findViewByid,onCclick等)
-
注解处理器需要实现AbstractProcessor接口,并实现对应的方法:init(),getSupportedSourceVersion(),getSupportedAnnotationTypes(),(返回所需要的处理的注解被process()方法接收), process() 方法必须实现,扫描所有备注接的元素并做处理,在这里完成了目标类信息的收集并生成对应 java 类,返回值类型是boolean,true表示该注解已被处理,不希望下一个注解处理器继续处理,false表示未被处理,下一个注解处理器继续处理
可用注解和反射实现
- 拿BindView为例:首先创建BindView注解,参数为默认 int value()
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.FIELD)
@interface BindView {
int value();
}
- 在相应的类中进行引用:
@BindView(R.id.text1)
TextView textView1;
@BindView(R.id.text2)
TextView textView2;
- 通过反射进行绑定 ButterKnife.bind(this);
public static void bind(Activity activity) {
bindView(activity);
}
//通过反射实现注解的绑定
public static void bindView(Activity activity) {
try {
// 获取字节码对象
Class<? extends Activity> aClass = activity.getClass();
// 获取全部变量
Field[] fields = aClass.getDeclaredFields();
// 遍历全部变量
for (Field field : fields) {
// 允许暴力反射
field.setAccessible(true);
// 获取带有注解BindView的变量
BindView annotation = field.getAnnotation(BindView.class);
if (annotation != null) {
// 获取注解的值
int value = annotation.value();
View view = activity.findViewById(value);
field.set(activity, view);
}
}
} catch (IllegalAccessException e) {
e.printStackTrace();
}
}