引言
在Android开发中我们会用到ButterKnife框架来简化绑定layout中的视图。这里我们主要分析ButterKnife框架中的流程。
简单使用
在Activity中使用
public class ButterKnifeDemo extends Activity {
@Bind(R.id.toolbar)
Toolbar toolbar;
@Bind(R.id.listView)
ListView listView;
@Bind(R.id.fab)
FloatingActionButton fab;
@Override
public void onCreate(Bundle savedInstance) {
super.onCreate(savedInstance);
setContentView(R.layout.activity_main);
ButterKnife.bind(this);
}
}
在非Activity中使用
View view = inflater.inflate(R.layout.fancy_fragment, container, false);
ButterKnife.bind(this, view);
这篇博客我们详细讲用法,只说原理,其他用法可以参考这篇博客
ButterKnife分析 产生疑问
@Bind(R.id.***)
TextView textView;//采用注解
注解:注解也称为元数据,为我们在代码中添加信息提供了一种形式化的方法,使我们可以在稍后某个时刻非常方便的使用这些数据。->简而言之就是换种方式定义数据。
如果想详细了解注解->传送门。
@Bind注解的定义:
@Retention(CLASS) @Target(FIELD)
public @interface Bind {
/** View ID to which the field will be bound. */
int[] value();
}
@Retention(CLASS) -> 注解在类文件中可用
@Target(FIELD) -> 注解可以用于类文件中的域(包括enum实例)
int[] value(); -> 定义属性value
注解写完后,就是绑定:
ButterKnife.bind(this);
我们来看下bind()源码:
看第二张图片你会发现:
Class<?> targetClass = target.getClass();
ViewBinder<Object> viewBinder = findViewBinderForClass(targetClass);
viewBinder.bind(finder, target, source);
获取到要绑定视图当前类,即Activity或者Fragment. 然后找到ViewBinder类并实例化进行绑定视图。
在这里你会产生疑问,我并没有在Activity或者Fragment中定义ViewBinder,哪里来的呢?
解决疑问
ButterKnife采用的是Java Annotation Processing 在编译时增加代码生成.class文件。与反射动态代理不同的是,一个是编译时,一个是运行时,性能当然是前者好。如果你要详细了解,请点击Java Annotation Processing
了解完Java Annotation Processing,查看ButterKnife,找到ButterKnife的核心代码就是ButterKnifeProcessor这个类。看下主要方法process.
@Override public boolean process(Set<? extends TypeElement> elements, RoundEnvironment env) {
Map<TypeElement, BindingClass> targetClassMap = findAndParseTargets(env);
for (Map.Entry<TypeElement, BindingClass> entry : targetClassMap.entrySet()) {
TypeElement typeElement = entry.getKey();
BindingClass bindingClass = entry.getValue();
try {
JavaFileObject jfo = filer.createSourceFile(bindingClass.getFqcn(), typeElement);
Writer writer = jfo.openWriter();
writer.write(bindingClass.brewJava());
writer.flush();
writer.close();
} catch (IOException e) {
error(typeElement, "Unable to write view binder for type %s: %s", typeElement,
e.getMessage());
}
}
return true;
}
解析下ButterKnife的工作流程:
- 扫描ButterKnife的注解
- 扫描的注解类中写入Java代码,即实现ViewBinder接口的类。
- 编译生成.class文件
- 然后程序运行执行.bind() method.
生成实现ViewBinder接口的类后,就和前面发出疑问的流程相符合。
总结
- Java有两种自动生成代码,一种是运行时即反射动态代理,将接口动态代理为具体的类。另外一种就是javac编译的工具Java Annoation Processer,在编译时生成java代码。当然后者性能上更好,只是为了方便开发者,前面则是用代理动态的生成代码,可以称之为无实际的java代码。