為什么要寫這一系列的博客呢?
創(chuàng)新互聯(lián)公司云計(jì)算的互聯(lián)網(wǎng)服務(wù)提供商,擁有超過13年的服務(wù)器租用、西信服務(wù)器托管、云服務(wù)器、網(wǎng)站空間、網(wǎng)站系統(tǒng)開發(fā)經(jīng)驗(yàn),已先后獲得國家工業(yè)和信息化部頒發(fā)的互聯(lián)網(wǎng)數(shù)據(jù)中心業(yè)務(wù)許可證。專業(yè)提供云主機(jī)、網(wǎng)站空間、域名注冊、VPS主機(jī)、云服務(wù)器、香港云服務(wù)器、免備案服務(wù)器等。
因?yàn)樵?Android 開發(fā)的過程中, 泛型,反射,注解這些知識進(jìn)場會用到,幾乎所有的框架至少都會用到上面的一兩種知識,如 Gson 就用到泛型,反射,注解,Retrofit 也用到泛型,反射,注解 。學(xué)好這些知識對我們進(jìn)階非常重要,尤其是閱讀開源框架源碼或者自己開發(fā)開源框架。
ButterKnife 這個開源庫火了有一段時(shí)間了,剛開始它的實(shí)現(xiàn)原理是使用反射實(shí)現(xiàn)的,性能較差。再后面的 版本中逐漸使用注解+放射實(shí)現(xiàn),性能提高了不少。
ButterKnife是基于編譯時(shí)的框架,它能夠幫助我們減去每次寫 FindViewById 的麻煩,截止到 2017.5.1 ,在 github 上面的 start 已經(jīng)超過 15000.
本篇博客要分析的 ButterKnife 的源碼主要包括以下三個部分,版本號是8.5.1
其中 butterknife-annotations 庫主要用來存放自定義注解;butterknife-compiler 主要是用來掃描哪些地方使用到我們的自定義注解,并進(jìn)行相應(yīng)的處理,生成模板代碼等;butterknife 主要是用來注入我們的代碼的。
我們先來先一下要怎樣使用 butterknife:
ButterKnife 的基本使用
在 moudle 的 build.gradle 增加依賴
dependencies {
compile 'com.jakewharton:butterknife:8.5.1'
annotationProcessor 'com.jakewharton:butterknife-compiler:8.5.1'
}
public class SimpleActivity extends Activity {
private static final ButterKnife.Action ALPHA_FADE = new ButterKnife.Action() {
@Override public void apply(@NonNull View view, int index) {
AlphaAnimation alphaAnimation = new AlphaAnimation(0, 1);
alphaAnimation.setFillBefore(true);
alphaAnimation.setDuration(500);
alphaAnimation.setStartOffset(index * 100);
view.startAnimation(alphaAnimation);
}
};
@BindView(R2.id.title) TextView title;
@BindView(R2.id.subtitle) TextView subtitle;
@BindView(R2.id.hello) Button hello;
@BindView(R2.id.list_of_things) ListView listOfThings;
@BindView(R2.id.footer) TextView footer;
@BindViews({ R2.id.title, R2.id.subtitle, R2.id.hello }) List headerViews;
private SimpleAdapter adapter;
@OnClick(R2.id.hello) void sayHello() {
Toast.makeText(this, "Hello, views!", LENGTH_SHORT).show();
ButterKnife.apply(headerViews, ALPHA_FADE);
}
@OnLongClick(R2.id.hello) boolean sayGetOffMe() {
Toast.makeText(this, "Let go of me!", LENGTH_SHORT).show();
return true;
}
@OnItemClick(R2.id.list_of_things) void onItemClick(int position) {
Toast.makeText(this, "You clicked: " + adapter.getItem(position), LENGTH_SHORT).show();
}
@Override protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.simple_activity);
ButterKnife.bind(this);
// Contrived code to use the bound fields.
title.setText("Butter Knife");
subtitle.setText("Field and method binding for Android views.");
footer.setText("by Jake Wharton");
hello.setText("Say Hello");
adapter = new SimpleAdapter(this);
listOfThings.setAdapter(adapter);
}
}
調(diào)用 gradle build 命令,我們在相應(yīng)的目錄下將可以看到生成類似這樣的代碼。
public class SimpleActivity_ViewBinding implements Unbinder {
protected T target;
private View view2130968578;
private View view2130968579;
@UiThread
public SimpleActivity_ViewBinding(final T target, View source) {
this.target = target;
View view;
target.title = Utils.findRequiredViewAsType(source, R.id.title, "field 'title'", TextView.class);
target.subtitle = Utils.findRequiredViewAsType(source, R.id.subtitle, "field 'subtitle'", TextView.class);
view = Utils.findRequiredView(source, R.id.hello, "field 'hello', method 'sayHello', and method 'sayGetOffMe'");
target.hello = Utils.castView(view, R.id.hello, "field 'hello'", Button.class);
view2130968578 = view;
view.setOnClickListener(new DebouncingOnClickListener() {
@Override
public void doClick(View p0) {
target.sayHello();
}
});
view.setOnLongClickListener(new View.OnLongClickListener() {
@Override
public boolean onLongClick(View p0) {
return target.sayGetOffMe();
}
});
view = Utils.findRequiredView(source, R.id.list_of_things, "field 'listOfThings' and method 'onItemClick'");
target.listOfThings = Utils.castView(view, R.id.list_of_things, "field 'listOfThings'", ListView.class);
view2130968579 = view;
((AdapterView>) view).setOnItemClickListener(new AdapterView.OnItemClickListener() {
@Override
public void onItemClick(AdapterView> p0, View p1, int p2, long p3) {
target.onItemClick(p2);
}
});
target.footer = Utils.findRequiredViewAsType(source, R.id.footer, "field 'footer'", TextView.class);
target.headerViews = Utils.listOf(
Utils.findRequiredView(source, R.id.title, "field 'headerViews'"),
Utils.findRequiredView(source, R.id.subtitle, "field 'headerViews'"),
Utils.findRequiredView(source, R.id.hello, "field 'headerViews'"));
}
@Override
@CallSuper
public void unbind() {
T target = this.target;
if (target == null) throw new IllegalStateException("Bindings already cleared.");
target.title = null;
target.subtitle = null;
target.hello = null;
target.listOfThings = null;
target.footer = null;
target.headerViews = null;
view2130968578.setOnClickListener(null);
view2130968578.setOnLongClickListener(null);
view2130968578 = null;
((AdapterView>) view2130968579).setOnItemClickListener(null);
view2130968579 = null;
this.target = null;
}
}
ButterKnife 的執(zhí)行流程
總的來說,大概可以分為以下幾步:
第一步:在編譯的時(shí)候掃描注解,并做相應(yīng)的處理,生成 java 代碼。這一步,可以拆分為幾個小步驟:
butterknife-annotations 講解
我們知道 ButterKnife 自定義很多的注解,有 BindArray,BindBitmap,BindColor,BindView 等,這里我們以 BindView 為例子講解就 OK 了,其他的也是基本類似的,這里就不再講解了。
//編譯時(shí)注解
@Retention(CLASS)
//成員變量, (includes enum constants)
@Target(FIELD)
public @interface BindView {
/** View ID to which the field will be bound. */
@IdRes int value();
}
Processor 解析器說明
我們先來看一些基本方法:在 init 方法里面得到一些輔助工具類,這樣有一個好處,確保工具類是單例的,因?yàn)?init 方法只會在初始化的時(shí)候調(diào)用。
public synchronized void init(ProcessingEnvironment env) {
super.init(env);
---
//輔助工具類
elementUtils = env.getElementUtils();
typeUtils = env.getTypeUtils();
filer = env.getFiler();
---
}
接著重寫 getSupportedAnnotationTypes 方法,返回我們支持的注解類型。
@Override
public Set getSupportedAnnotationTypes() {
Set types = new LinkedHashSet<>();
for (Class extends Annotation> annotation : getSupportedAnnotations()) {
types.add(annotation.getCanonicalName());
}
//返回支持注解的類型
return types;
}
private Set> getSupportedAnnotations() {
Set> annotations = new LinkedHashSet<>();
annotations.add(BindArray.class);
annotations.add(BindBitmap.class);
annotations.add(BindBool.class);
annotations.add(BindColor.class);
annotations.add(BindDimen.class);
annotations.add(BindDrawable.class);
annotations.add(BindFloat.class);
annotations.add(BindInt.class);
annotations.add(BindString.class);
annotations.add(BindView.class);
annotations.add(BindViews.class);
annotations.addAll(LISTENERS);
return annotations;
}
接下來來看我們的重點(diǎn), process 方法。所做的工作大概就是拿到我們所有的注解信息,存進(jìn) map 集合,遍歷 map 集合,做相應(yīng)的 處理,生成 java 代碼。
@Override
public boolean process(Set extends TypeElement> elements, RoundEnvironment env) {
// 拿到所有的注解信息,TypeElement 作為 key,BindingSet 作為 value
Map bindingMap = findAndParseTargets(env);
// 遍歷 map 里面的所有信息,并生成 java 代碼
for (Map.Entry entry : bindingMap.entrySet()) {
TypeElement typeElement = entry.getKey();
BindingSet binding = entry.getValue();
JavaFile javaFile = binding.brewJava(sdk);
try {
javaFile.writeTo(filer);
} catch (IOException e) {
error(typeElement, "Unable to write binding for type %s: %s", typeElement, e
.getMessage());
}
}
return false;
}
這里我們進(jìn)入 findAndParseTargets 方法,看里面到底是怎樣將注解信息存進(jìn) map 集合的?
findAndParseTargets 方法里面 針對每一個自定義注解(BindArray,BindBitmap,BindColor,BindView) 等都做了處理,這里我們重點(diǎn)關(guān)注 @BindView 的處理即可。其他注解的處理思想也是一樣的。
我們先來看一下 findAndParseTargets 方法的前半部分,遍歷 env.getElementsAnnotatedWith(BindView.class) 集合,并調(diào)用 parseBindView 方法去轉(zhuǎn)化。
private Map findAndParseTargets(RoundEnvironment env) {
Map builderMap = new LinkedHashMap<>();
Set erasedTargetNames = new LinkedHashSet<>();
scanForRClasses(env);
// Process each @BindView element.
for (Element element : env.getElementsAnnotatedWith(BindView.class)) {
// we don't SuperficialValidation.validateElement(element)
// so that an unresolved View type can be generated by later processing rounds
try {
parseBindView(element, builderMap, erasedTargetNames);
} catch (Exception e) {
logParsingError(element, BindView.class, e);
}
}
---
// 后半部分,待會再講
}
可以看到牽絆部分的主要邏輯在 parseBindView 方法里面,主要做了以下幾步操作:
判斷被注解 @BindView 修飾的成員變量是不是合法的,private 或者 static 修飾的,則出錯。
private void parseBindView(Element element, Map builderMap,
Set erasedTargetNames) {
TypeElement enclosingElement = (TypeElement) element.getEnclosingElement();
// 判斷是否被注解在屬性上,如果該屬性是被 private 或者 static 修飾的,則出錯
// 判斷是否被注解在錯誤的包中,若包名以“android”或者“java”開頭,則出錯
boolean hasError = isInaccessibleViaGeneratedCode(BindView.class, "fields", element)
|| isBindingInWrongPackage(BindView.class, element);
// Verify that the target type extends from View.
TypeMirror elementType = element.asType();
if (elementType.getKind() == TypeKind.TYPEVAR) {
TypeVariable typeVariable = (TypeVariable) elementType;
elementType = typeVariable.getUpperBound();
}
Name qualifiedName = enclosingElement.getQualifiedName();
Name simpleName = element.getSimpleName();
// 判斷元素是不是View及其子類或者Interface
if (!isSubtypeOfType(elementType, VIEW_TYPE) && !isInterface(elementType)) {
if (elementType.getKind() == TypeKind.ERROR) {
note(element, "@%s field with unresolved type (%s) "
+ "must elsewhere be generated as a View or interface. (%s.%s)",
BindView.class.getSimpleName(), elementType, qualifiedName, simpleName);
} else {
error(element, "@%s fields must extend from View or be an interface. (%s.%s)",
BindView.class.getSimpleName(), qualifiedName, simpleName);
hasError = true;
}
}
// 如果有錯誤,直接返回
if (hasError) {
return;
}
// Assemble information on the field.
int id = element.getAnnotation(BindView.class).value();
// 根據(jù)所在的類元素去查找 builder
BindingSet.Builder builder = builderMap.get(enclosingElement);
QualifiedId qualifiedId = elementToQualifiedId(element, id);
// 如果相應(yīng)的 builder 已經(jīng)存在
if (builder != null) {
// 驗(yàn)證 ID 是否已經(jīng)被綁定
String existingBindingName = builder.findExistingBindingName(getId(qualifiedId));
// 被綁定了,出錯,返回
if (existingBindingName != null) {
error(element, "Attempt to use @%s for an already bound ID %d on '%s'. (%s.%s)",
BindView.class.getSimpleName(), id, existingBindingName,
enclosingElement.getQualifiedName(), element.getSimpleName());
return;
}
} else {
// 如果沒有相應(yīng)的 builder,就需要重新生成,并別存放到 builderMap 中
builder = getOrCreateBindingBuilder(builderMap, enclosingElement);
}
String name = simpleName.toString();
TypeName type = TypeName.get(elementType);
boolean required = isFieldRequired(element);
builder.addField(getId(qualifiedId), new FieldViewBinding(name, type, required));
// Add the type-erased version to the valid binding targets set.
erasedTargetNames.add(enclosingElement);
}
parseBindView 方法分析完畢之后,我們在回過頭來看一下 findAndParseTargets 方法的后半部分,主要做的工作是對 bindingMap 進(jìn)行重排序。
private Map findAndParseTargets(RoundEnvironment env) {
// 省略前半部分
// Associate superclass binders with their subclass binders. This is a queue-based tree walk
// which starts at the roots (superclasses) and walks to the leafs (subclasses).
Deque> entries =
new ArrayDeque<>(builderMap.entrySet());
Map bindingMap = new LinkedHashMap<>();
while (!entries.isEmpty()) {
Map.Entry entry = entries.removeFirst();
TypeElement type = entry.getKey();
BindingSet.Builder builder = entry.getValue();
//獲取 type 的父類的 TypeElement
TypeElement parentType = findParentType(type, erasedTargetNames);
// 為空,存進(jìn) map
if (parentType == null) {
bindingMap.put(type, builder.build());
} else {
// 獲取 parentType 的 BindingSet
BindingSet parentBinding = bindingMap.get(parentType);
if (parentBinding != null) {
builder.setParent(parentBinding);
bindingMap.put(type, builder.build());
} else {
// Has a superclass binding but we haven't built it yet. Re-enqueue for later.
// 為空,加到隊(duì)列的尾部,等待下一次處理
entries.addLast(entry);
}
}
}
return bindingMap;
}
到這里為止,我們已經(jīng)分析完 ButterKnifeProcessor 是怎樣處理注解的相關(guān)知識,并存進(jìn) map 集合中的,下面我們回到 process 方法,看一下是怎樣生成 java 模板代碼的。
public boolean process(Set extends TypeElement> elements, RoundEnvironment env) {
// 拿到所有的注解信息,TypeElement 作為 key,BindingSet 作為 value
Map bindingMap = findAndParseTargets(env);
// 遍歷 map 里面的所有信息,并生成 java 代碼
for (Map.Entry entry : bindingMap.entrySet()) {
TypeElement typeElement = entry.getKey();
BindingSet binding = entry.getValue();
// 生成 javaFile 對象
JavaFile javaFile = binding.brewJava(sdk);
try {
// 生成 java 模板代碼
javaFile.writeTo(filer);
} catch (IOException e) {
error(typeElement, "Unable to write binding for type %s: %s", typeElement, e
.getMessage());
}
}
return false;
}
生成代碼的核心代碼只有這幾行
// 生成 javaFile 對象
JavaFile javaFile = binding.brewJava(sdk);
try {
// 生成 java 模板代碼
javaFile.writeTo(filer);
} catch (IOException e) {
error(typeElement, "Unable to write binding for type %s: %s", typeElement, e
.getMessage());
}
跟蹤進(jìn)去,發(fā)現(xiàn)是調(diào)用 square 公司開源的庫 javapoet 開生成代碼的。關(guān)于 javaPoet 的使用可以參考官網(wǎng)地址
JavaFile brewJava(int sdk) {
return JavaFile.builder(bindingClassName.packageName(), createType(sdk))
.addFileComment("Generated code from Butter Knife. Do not modify!")
.build();
}
private TypeSpec createType(int sdk) {
TypeSpec.Builder result = TypeSpec.classBuilder(bindingClassName.simpleName())
.addModifiers(PUBLIC);
if (isFinal) {
result.addModifiers(FINAL);
}
if (parentBinding != null) {
result.superclass(parentBinding.bindingClassName);
} else {
result.addSuperinterface(UNBINDER);
}
if (hasTargetField()) {
result.addField(targetTypeName, "target", PRIVATE);
}
// 如果是 View 或者是 View 的子類的話,添加構(gòu)造方法
if (isView) {
result.addMethod(createBindingConstructorForView());
} else if (isActivity) { // 如果是 Activity 或者是 Activity 的子類的話,添加構(gòu)造方法
result.addMethod(createBindingConstructorForActivity());
} else if (isDialog) { // 如果是 Dialog 或者是 Dialog 的子類的話,添加構(gòu)造方法
result.addMethod(createBindingConstructorForDialog());
}
// 如果構(gòu)造方法不需要 View 參數(shù),添加 需要 View 參數(shù)的構(gòu)造方法
if (!constructorNeedsView()) {
// Add a delegating constructor with a target type + view signature for reflective use.
result.addMethod(createBindingViewDelegateConstructor());
}
result.addMethod(createBindingConstructor(sdk));
if (hasViewBindings() || parentBinding == null) {
//生成unBind方法
result.addMethod(createBindingUnbindMethod(result));
}
return result.build();
}
接著我們一起來看一下 createBindingConstructor(sdk) 方法,大概做的事情就是
- 判斷是否有設(shè)置監(jiān)聽,如果有監(jiān)聽,將 View 設(shè)置為 final
- 遍歷 viewBindings ,調(diào)用 addViewBinding 生成 findViewById 形式的代碼。
private MethodSpec createBindingConstructor(int sdk) {
MethodSpec.Builder constructor = MethodSpec.constructorBuilder()
.addAnnotation(UI_THREAD)
.addModifiers(PUBLIC);
// 如果有方法綁定,比如 @onClick,那么增加一個 targetTypeName 類型 的方法參數(shù) target,并且是 final 類型的
if (hasMethodBindings()) {
constructor.addParameter(targetTypeName, "target", FINAL);
} else { // 如果沒有 ,不是 final 類型的
constructor.addParameter(targetTypeName, "target");
}
//如果有注解的 View,那么添加 VIEW 類型 source 參數(shù)
if (constructorNeedsView()) {
constructor.addParameter(VIEW, "source");
} else {
// 添加 Context 類型的 context 參數(shù)
constructor.addParameter(CONTEXT, "context");
}
if (hasUnqualifiedResourceBindings()) {
// Aapt can change IDs out from underneath us, just suppress since all will work at
// runtime.
constructor.addAnnotation(AnnotationSpec.builder(SuppressWarnings.class)
.addMember("value", "$S", "ResourceType")
.build());
}
// 如果 @OnTouch 綁定 View,添加 @SuppressLint("ClickableViewAccessibility")
if (hasOnTouchMethodBindings()) {
constructor.addAnnotation(AnnotationSpec.builder(SUPPRESS_LINT)
.addMember("value", "$S", "ClickableViewAccessibility")
.build());
}
// 如果 parentBinding 不為空,調(diào)用父類 的構(gòu)造方法
if (parentBinding != null) {
if (parentBinding.constructorNeedsView()) {
constructor.addStatement("super(target, source)");
} else if (constructorNeedsView()) {
constructor.addStatement("super(target, source.getContext())");
} else {
constructor.addStatement("super(target, context)");
}
constructor.addCode("\n");
}
// 添加成員變量
if (hasTargetField()) {
constructor.addStatement("this.target = target");
constructor.addCode("\n");
}
if (hasViewBindings()) {
if (hasViewLocal()) {
// Local variable in which all views will be temporarily stored.
constructor.addStatement("$T view", VIEW);
}
// 遍歷 viewBindings,生成 source.findViewById($L) 代碼
for (ViewBinding binding : viewBindings) {
addViewBinding(constructor, binding);
}
for (FieldCollectionViewBinding binding : collectionBindings) {
constructor.addStatement("$L", binding.render());
}
if (!resourceBindings.isEmpty()) {
constructor.addCode("\n");
}
}
if (!resourceBindings.isEmpty()) {
if (constructorNeedsView()) {
constructor.addStatement("$T context = source.getContext()", CONTEXT);
}
if (hasResourceBindingsNeedingResource(sdk)) {
constructor.addStatement("$T res = context.getResources()", RESOURCES);
}
for (ResourceBinding binding : resourceBindings) {
constructor.addStatement("$L", binding.render(sdk));
}
}
return constructor.build();
}
下面我們一起來看一下 addViewBinding 方法是怎樣生成代碼的。
private void addViewBinding(MethodSpec.Builder result, ViewBinding binding) {
if (binding.isSingleFieldBinding()) {
// Optimize the common case where there's a single binding directly to a field.
FieldViewBinding fieldBinding = binding.getFieldBinding();
// 注意這里直接使用了 target. 的形式,所以屬性肯定是不能 private 的
CodeBlock.Builder builder = CodeBlock.builder()
.add("target.$L = ", fieldBinding.getName());
boolean requiresCast = requiresCast(fieldBinding.getType());
if (!requiresCast && !fieldBinding.isRequired()) {
builder.add("source.findViewById($L)", binding.getId().code);
} else {
builder.add("$T.find", UTILS);
builder.add(fieldBinding.isRequired() ? "RequiredView" : "OptionalView");
if (requiresCast) {
builder.add("AsType");
}
builder.add("(source, $L", binding.getId().code);
if (fieldBinding.isRequired() || requiresCast) {
builder.add(", $S", asHumanDescription(singletonList(fieldBinding)));
}
if (requiresCast) {
builder.add(", $T.class", fieldBinding.getRawType());
}
builder.add(")");
}
result.addStatement("$L", builder.build());
return;
}
**ButterKnife 是怎樣實(shí)現(xiàn)代碼注入的**
使用過 ButterKnife 得人基本都知道,我們是通過 bind 方法來實(shí)現(xiàn)注入的,即自動幫我們 findViewById ,解放我們的雙手,提高工作效率。下面我們一起來看一下 bind 方法是怎樣實(shí)現(xiàn)注入的。
@NonNull@UiThread
br/>@UiThread
View sourceView = target.getWindow().getDecorView();
return createBinding(target, sourceView);
}
可以看到 bind 方法很簡單,邏輯基本都交給 createBinding 方法去完成。我們一起進(jìn)入 createBinding 方法來看一下到底做了什么
private static Unbinder createBinding(@NonNull Object target, @NonNull View source) {
Class> targetClass = target.getClass();
if (debug) Log.d(TAG, "Looking up binding for " + targetClass.getName());
// 從 Class 中查找 constructor
Constructor extends Unbinder> constructor = findBindingConstructorForClass(targetClass);
if (constructor == null) {
return Unbinder.EMPTY;
}
//noinspection TryWithIdenticalCatches Resolves to API 19+ only type.
try {
// 反射實(shí)例化構(gòu)造方法
return constructor.newInstance(target, source);
} catch (IllegalAccessException e) {
throw new RuntimeException("Unable to invoke " + constructor, e);
} catch (InstantiationException e) {
throw new RuntimeException("Unable to invoke " + constructor, e);
} catch (InvocationTargetException e) {
Throwable cause = e.getCause();
if (cause instanceof RuntimeException) {
throw (RuntimeException) cause;
}
if (cause instanceof Error) {
throw (Error) cause;
}
throw new RuntimeException("Unable to create binding instance.", cause);
}
}
其實(shí) createBinding 來說,主要做了這幾件事情
- 傳入 class ,通過 findBindingConstructorForClass 方法來實(shí)例化 constructor
- 利用反射來初始化 constructor 對象
- 初始化 constructor 失敗會拋出異常
下面我們一起來看一下 findBindingConstructorForClass 方法是怎樣實(shí)現(xiàn)的。
private static Constructor extends Unbinder> findBindingConstructorForClass(Class> cls) {
// 讀取緩存,如果不為空,直接返回
Constructor extends Unbinder> bindingCtor = BINDINGS.get(cls);
if (bindingCtor != null) {
if (debug) Log.d(TAG, "HIT: Cached in binding map.");
return bindingCtor;
}
// 如果是 android ,java 原生的文件,不處理
String clsName = cls.getName();
if (clsName.startsWith("android.") || clsName.startsWith("java.")) {
if (debug) Log.d(TAG, "MISS: Reached framework class. Abandoning search.");
return null;
}
try {
Class> bindingClass = cls.getClassLoader().loadClass(clsName + "_ViewBinding");
//noinspection unchecked
// 在原來所在的類查找
bindingCtor = (Constructor extends Unbinder>) bindingClass.getConstructor(cls, View
.class);
if (debug) Log.d(TAG, "HIT: Loaded binding class and constructor.");
} catch (ClassNotFoundException e) {
if (debug) Log.d(TAG, "Not found. Trying superclass " + cls.getSuperclass().getName());
// 在原來的類查找,查找不到,到父類去查找
bindingCtor = findBindingConstructorForClass(cls.getSuperclass());
} catch (NoSuchMethodException e) {
throw new RuntimeException("Unable to find binding constructor for " + clsName, e);
}
// 存進(jìn) LinkedHashMap 緩存
BINDINGS.put(cls, bindingCtor);
return bindingCtor;
}
它的實(shí)現(xiàn)思想是這樣的:
- 讀取緩存,若緩存命中,直接返回,這樣有利于提高效率。從代碼中可以看到,緩存是通過存進(jìn) map 集合實(shí)現(xiàn)的。
- 是否是我們目標(biāo)文件,是的話,進(jìn)行處理,不是的話,直接返回,并打印相應(yīng)的日志
- 利用類加載器加載我們自己生成的 class 文件,并獲取其構(gòu)造方法,獲取到,直接返回。獲取不到,會拋出異常,在異常的處理中,我們再從當(dāng)前 class 文件的父類去查找。并把結(jié)果存進(jìn) map 集合中,做緩存處理。
我們對 ButterKnife 的分析到此為止。
**題外話**
這篇博客主要是分析了 ButterKnife 的主要原理實(shí)現(xiàn),對 ButterKnife 里面的一些實(shí)現(xiàn)細(xì)節(jié)并未詳細(xì)分析。不過對我們讀懂代碼已經(jīng)足夠了。下一個系列,主要講解 CoordinatorLayout 的實(shí)現(xiàn)原理及怎樣自定義 CoordinatorLayout 的 behavior 實(shí)現(xiàn)仿新浪微博發(fā)現(xiàn)頁面的效果,敬請期待。
**關(guān)于我**
更多信息可以點(diǎn)擊[關(guān)于我](https://www.jianshu.com/p/78f6f0b5bce4)
, 非常希望和大家一起交流 , 共同進(jìn)步