ButterKnife实现原理

代码自动生成

  • List item使用代码自动生成,一是为了提高编码的效率,二十避免在运行期大量使用反射,通过在编译期利用反射生成辅助类和方法以供运行时使用

注解处理器步骤

  • 在java编译器中构建
  • 编译器开始执行未执行过的注解处理器
  • 玄幻处理注解元素,找到被该住解所修饰的类,方法或者属性
  • 生成对应的类,并写入文件
  • 判断是否所有的注解处理器都已执行完毕,如果没有,继续下一个注解处理器的执行(回到步骤1)

ButterKnife实现原理

注解处理器的实现

  • 主要需要两个库的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();
    }
}
上一篇:C++前向引用声明


下一篇:android-studio-2021.1.1.22-windows.exe“Android-ButterKnife-Injections“is incompatible解决过程记录