5. Android 框架ButterKnife源代码分析

简介:

一. ButterKnife介绍

在Android编程过程中,我们会写大量的布局和点击事件,像初始view、设置view监听这样简单而重复的操作,这些代码繁琐而又不雅观,比如:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
TextView tvSetName = findViewById(R.id.xxx);
tvSetName.setOnClickListener( new  View.OnClickListener() {
     @Override
     public  void  onClick(View v) {
         //xxxx
     }
});
TextView tvSetAge = findViewById(R.id.xxx);
tvSetAge.setOnClickListener( new  View.OnClickListener() {
     @Override
     public  void  onClick(View v) {
         //xxxx
     }
});
TextView tvSetArea = findViewById(R.id.xxx);
tvSetArea.setOnClickListener( new  View.OnClickListener() {
     @Override
     public  void  onClick(View v) {
         //xxxx
     }
});



Activity中这种代码多了之后,很不雅观。


二. 使用简介

ButterKnife使用方法比较简单,主要包括以下步骤:


  1. 引用ButterKnife包

  2. 在onCreate里面bind(setContentView之后)

  3. 绑定各种事件

  4. onDestroy里面解绑释放资源


ButterKnife使用方法


三. ButterKnife源代码下载


ButterKnife github源代码地址


直接git clone或者下载zip即可。


四. 编译

  1. Android studio打开ButterKnife源代码

    AndroidStudio->File->open->ButterKnife源代码路径->确认

  2. Build->Rebuild Project



五. 生成的aar和jar包

生成的包主要有两个

  1. butterknife-annotations-8.5.2-SNAPSHOT.jar

    路径:butterknife-annotations->build->libs


  2. butterknife-release.aar

    路径: butterknife->build->outputs->aar



六. 其他应用引用自定义ButterKnife包 

  1. 删除原来ButterKnife包引用,因为要使用自己编译的包

  2. //    compile 'com.jakewharton:butterknife:8.4.0'




  3. 拷贝文件

    拷贝上面两个文件到自己项目app模块的libs 目录


  4. 添加aar的关联

    打开app模块的build.gradle文件,添加:


  5. 1
    2
    compile fileTree(include: ['*.jar'], dir: 'libs')
    compile(name: 'butterknife-release', ext: 'aar')
  6. Build->Rebuild Project



七. 源代码分析


1. ButterKnife.bind(Activity target)过程


文件名:ButterKnife.java

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
static  final  Map<Class<?>, Constructor<?  extends  Unbinder>> BINDINGS =  new  LinkedHashMap<>();
 
public  static  Unbinder bind( @NonNull  Activity target) {
     Log.d( "Sandy" "ButterKnife bind.. target: "  + target);
     View sourceView = target.getWindow().getDecorView();
     return  createBinding(target, sourceView);
}
 
 
 
 
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());
     Constructor<?  extends  Unbinder> constructor = findBindingConstructorForClass(targetClass);
 
     if  (constructor ==  null ) {
       return  Unbinder.EMPTY;
     }
 
     //noinspection TryWithIdenticalCatches Resolves to API 19+ only type.
     try  {
       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);
     }
}
 
 
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;
     }
     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  {
         Log.d( "Sandy" "findBindingConsForClass: "  + clsName +  " vindBinding name: "  +
                 clsName +  "_ViewBinding" );
       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);
     }
     BINDINGS.put(cls, bindingCtor);
     return  bindingCtor;
}




上面这段代码有几个注意点:

a. sourceView代表是DecorView,也就是我们窗口的顶级View。


b. findBindingConstructorForClass有个BINDINGS缓存,key是class,value是缓存的Unbinder对象,这样做可以加快bind速度。

因为每个类的ButterKnife注解在运行期间是不会变的,比如MainActivity有3个ButterKnife注解,那么它就是3个。除非有新的apk安装。

所以适合用缓存来实现。


c. findBindingConstructorForClass使用了递归的方法

这个方法使用了递归,不断调用父类,也就是


1
2
3
4
catch  (ClassNotFoundException e) {
       if  (debug) Log.d(TAG,  "Not found. Trying superclass "  + cls.getSuperclass().getName());
       bindingCtor = findBindingConstructorForClass(cls.getSuperclass());
}



那为什么要这么处理呢?


因为有些Activity没有ButterKnife的注解,但是它的父类可能有,比如BaseActivity。所以需要往上递归。那什么时候递归结束呢?


1
2
3
4
5
6
7
8
9
if  (bindingCtor !=  null ) {
       if  (debug) Log.d(TAG,  "HIT: Cached in binding map." );
       return  bindingCtor;
     }
     String clsName = cls.getName();
     if  (clsName.startsWith( "android." ) || clsName.startsWith( "java." )) {
       if  (debug) Log.d(TAG,  "MISS: Reached framework class. Abandoning search." );
       return  null ;
}



如果缓存里面找到了结果,那么结束,同时返回结果;


或者类名以"android."或者"java."开头,也结束,返回null;

以Activity为例,Activity的类名是android.app.Activity,所以你的MainActivity如果递归到Activity还没有找到ButterKnife注解,那就说明你的MainActivity是没有包含ButterKnife注解的。


d. 如果子Activity和父Activity都有ButterKnife注解怎么办?

答案是返回子Activity以及其对应的 Constructor<? extends Unbinder> bindingCtor对象


那它的父Activity如果也有ButterKnife注解怎么办?怎么解析父Activity的ButterKnife注解呢? 这个问题我们待会再讲。

记为问题1


e. Constructor<? extends Unbinder> 是个什么东西?

调用ButterKnife.bind(Activity target)方法后会返回一个Unbinder对象,可以在onDestroy中调用unbind()方法,那个Unbinder是什么东西呢?这个问题待会再讲。

记为问题2.


f. clsName + "_ViewBinding"是什么类?

Class<?> bindingClass = cls.getClassLoader().loadClass(clsName + "_ViewBinding");

这个问题记为问题3.


2. ButterKnifeProcessor.java

这个类是ButterKnife里面很重要的一个类了,它继承自AbstractProcessor。


看来不懂的问题越来越多,那么有必要来学习下Java注解的知识。



八. Java注解


在分析ButterKnife代码前,需要了解Java的注解,需要了解Annotation Processor,因为ButterKnifer用到这个知识。


这里面很重要的一个知识点就是你可以编写一定的规则,让它在应用程序编译时执行你的规则,然后生成Java代码;并且生成的Java还可以参与编译。


Java注解



九. 自己定义的注解框架


1. Eclipse实现


主要是参考这篇帖子完成的,大家可以参考这篇帖子:

Eclipse中使用Java注解Processor


主要说下不同的地方:

a. source folder的创建,直接File->New->source folder一直创建不成功,后面用另外一种方法创建成功了。

项目->右击->Properties->Java Build Path->Source->Add Folder->Create New Folder->输入resources/META-INF/services->finish->ok->ok

wKiom1kUG_CSP3OvAACTzaqjj0w123.png-wh_50



2. Android studio实现


AndroidStudio下面使用Java注解Processor




十. 调试自己的自定义框架

有个时候需要调试自己写的框架是否正常运行,下面介绍下调试:


1. 在项目gradle.properties里面添加

1
2
org.gradle.daemon=true
org.gradle.jvmargs=-agentlib:jdwp=transport=dt_socket,server=y,suspend=n,address=8011


2.Edit Configureations

wKioL1kUUgXzetkqAADJBDdHWVg272.png-wh_50


3. 增加远程调试

wKiom1kUUlWiJR1iAADIV1Xx2G0912.png-wh_50


4. 启动远程调试

wKioL1kUUqSykGMqAAAtwfDm2S0203.png-wh_50



下面的控制台会出现下面的提示:

Connected to the target VM, address: 'localhost:8011', transport: 'socket'


5. 打断点

在Processor里面打上断点,比如init, process


6. 连上手机,项目根目录命令行下执行

gradle clean connectedCheck



十一. ButterKnife使用Java注解

理解了Java注解Processor之后,ButterKnife就比较好理解了。


首先它的ButterKnifeProcessor.java继承自AbstractProcessor,重写了init和process之类的方法。

也就是说它在编译的时候会被执行,生成辅助代码。


它的辅助代码生成到哪里了呢?


在我们自己的应用程序里面搜索_ViewBinding,就可以找到已经生成好的辅助类,如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
public  class  xxxx_ViewBinding<T  extends  LoginCloudActivity>  extends  BaseActivity_ViewBinding<T> {
   private  View view2131689593;
 
   @UiThread
   public  xxxx_ViewBinding( final  T target, View source) {
     super (target, source);
 
     View view;
     target.mEtUser = Utils.findRequiredViewAsType(source, R.id.et_user,  "field 'mEtUser'" , EditText. class );
     target.mEtPwd = Utils.findRequiredViewAsType(source, R.id.et_pwd,  "field 'mEtPwd'" , EditText. class );
     view = Utils.findRequiredView(source, R.id.btn_login,  "method 'btn_login' and method 'btn_login_long'" );
     view2131689593 = view;
     view.setOnClickListener( new  DebouncingOnClickListener() {
       @Override
       public  void  doClick(View p0) {
         target.btn_login();
       }
     });
     view.setOnLongClickListener( new  View.OnLongClickListener() {
       @Override
       public  boolean  onLongClick(View p0) {
         return  target.btn_login_long();
       }
     });
   }
 
   @Override
   public  void  unbind() {
     T target =  this .target;
     super .unbind();
 
     target.mEtUser =  null ;
     target.mEtPwd =  null ;
 
     view2131689593.setOnClickListener( null );
     view2131689593.setOnLongClickListener( null );
     view2131689593 =  null ;
   }
}


这个类在编译的时候会被自动生成,那么在运行的时候,它会被调用。

这个类的构造函数会去初始化那些控件,设置监听。


回到第七步 ButterKnife.bind()的过程


在createBinding的时候,它会初始化这个xxx_ViewBinding类,如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
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());
     Constructor<?  extends  Unbinder> constructor = findBindingConstructorForClass(targetClass);
 
     if  (constructor ==  null ) {
       return  Unbinder.EMPTY;
     }
 
     //noinspection TryWithIdenticalCatches Resolves to API 19+ only type.
     try  {
       return  constructor.newInstance(target, source);
       ...
       
   }


那么就会走到xxx_ViewBinding的构造函数,那么就会初始化控件,同时也会设置监听。如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
target.mEtUser = Utils.findRequiredViewAsType(source, R.id.et_user,  "field 'mEtUser'" , EditText. class );
     target.mEtPwd = Utils.findRequiredViewAsType(source, R.id.et_pwd,  "field 'mEtPwd'" , EditText. class );
     view = Utils.findRequiredView(source, R.id.btn_login,  "method 'btn_login' and method 'btn_login_long'" );
     view2131689593 = view;
     view.setOnClickListener( new  DebouncingOnClickListener() {
       @Override
       public  void  doClick(View p0) {
         target.btn_login();
       }
     });
     view.setOnLongClickListener( new  View.OnLongClickListener() {
       @Override
       public  boolean  onLongClick(View p0) {
         return  target.btn_login_long();
       }
     });



它的调用方式直接是target.mEtpwd,所以也就是说Activity的mEtpwd控件不能是private的,否则会引用不到。



参考网址:

butterknife github源代码下载


ButterKnife源代码解析


Java注解处理器分析


Eclipse中使用Java注解处理器


Android studio使用java注解处理器


JavaPoet介绍


调试Java注解处理器出错





     本文转自rongwei84n 51CTO博客,原文链接: http://blog.51cto.com/483181/1924089 ,如需转载请自行联系原作者


相关文章
|
10天前
|
物联网 区块链 vr&ar
未来已来:探索区块链、物联网与虚拟现实技术的融合与应用安卓与iOS开发中的跨平台框架选择
【8月更文挑战第30天】在科技的巨轮下,新技术不断涌现,引领着社会进步。本文将聚焦于当前最前沿的技术——区块链、物联网和虚拟现实,探讨它们各自的发展趋势及其在未来可能的应用场景。我们将从这些技术的基本定义出发,逐步深入到它们的相互作用和集成应用,最后展望它们如何共同塑造一个全新的数字生态系统。
|
1天前
|
开发工具 Android开发 Swift
安卓与iOS开发环境对比分析
在移动应用开发的广阔舞台上,安卓和iOS这两大操作系统无疑是主角。它们各自拥有独特的特点和优势,为开发者提供了不同的开发环境和工具。本文将深入浅出地探讨安卓和iOS开发环境的主要差异,包括开发工具、编程语言、用户界面设计、性能优化以及市场覆盖等方面,旨在帮助初学者更好地理解两大平台的开发特点,并为他们选择合适的开发路径提供参考。通过比较分析,我们将揭示不同环境下的开发实践,以及如何根据项目需求和目标受众来选择最合适的开发平台。
12 2
|
7天前
|
IDE 开发工具 Android开发
安卓与iOS开发环境对比分析
本文将探讨安卓和iOS这两大移动操作系统在开发环境上的差异,从工具、语言、框架到生态系统等多个角度进行比较。我们将深入了解各自的优势和劣势,并尝试为开发者提供一些实用的建议,以帮助他们根据自己的需求选择最适合的开发平台。
16 1
|
20天前
|
Java 开发工具 Android开发
安卓与iOS开发环境对比分析
【8月更文挑战第20天】在移动应用开发的广阔天地中,Android和iOS两大平台各自占据着重要的位置。本文将深入探讨这两种操作系统的开发环境,从编程语言到开发工具,从用户界面设计到性能优化,以及市场趋势对开发者选择的影响。我们旨在为读者提供一个全面的比较视角,帮助理解不同平台的优势与挑战,并为那些站在选择十字路口的开发者提供有价值的参考信息。
|
11天前
|
设计模式 Java Android开发
探索安卓应用开发:从新手到专家的旅程探索iOS开发中的SwiftUI框架
【8月更文挑战第29天】本文旨在通过一个易于理解的旅程比喻,带领读者深入探讨安卓应用开发的各个方面。我们将从基础概念入手,逐步过渡到高级技术,最后讨论如何维护和推广你的应用。无论你是编程新手还是有经验的开发者,这篇文章都将为你提供有价值的见解和实用的代码示例。让我们一起开始这段激动人心的旅程吧!
|
12天前
|
Android开发
基于Amlogic 安卓9.0, 驱动简说(三):使用misc框架,让驱动更简单
如何使用Amlogic T972安卓9.0系统上的misc框架来简化驱动程序开发,通过misc框架自动分配设备号并创建设备文件,从而减少代码量并避免设备号冲突。
18 0
基于Amlogic 安卓9.0, 驱动简说(三):使用misc框架,让驱动更简单
|
12天前
|
搜索推荐 Android开发
学习AOSP安卓系统源代码,需要什么样的电脑?不同配置的电脑,其编译时间有多大差距?
本文分享了不同价位电脑配置对于编译AOSP安卓系统源代码的影响,提供了从6000元到更高价位的电脑配置实例,并比较了它们的编译时间,以供学习AOSP源代码时电脑配置选择的参考。
32 0
学习AOSP安卓系统源代码,需要什么样的电脑?不同配置的电脑,其编译时间有多大差距?
|
20天前
|
开发框架 Android开发 Swift
安卓与iOS应用开发对比分析
【8月更文挑战第20天】在移动应用开发的广阔天地中,安卓和iOS两大平台各占半壁江山。本文将深入探讨这两大操作系统在开发环境、编程语言、用户界面设计、性能优化及市场分布等方面的差异和特点。通过比较分析,旨在为开发者提供一个宏观的视角,帮助他们根据项目需求和目标受众选择最合适的开发平台。同时,文章还将讨论跨平台开发框架的利与弊,以及它们如何影响着移动应用的开发趋势。
|
20天前
|
安全 搜索推荐 Android开发
安卓与iOS应用开发的对比分析
【8月更文挑战第20天】在移动应用开发领域,安卓和iOS两大平台各领风骚。本文通过深入探讨两者的开发环境、编程语言、用户界面设计、应用市场及分发机制等方面的差异,揭示了各自的优势和挑战。旨在为开发者提供决策支持,同时帮助理解为何某些应用可能优先选择在一个平台上发布。
25 2
|
24天前
|
存储 前端开发 Java
Android MVVM框架详解与应用
在Android开发中,随着应用复杂度的增加,如何有效地组织和管理代码成为了一个重要的问题。MVVM(Model-View-ViewModel)架构模式因其清晰的结构和高效的开发效率,逐渐成为Android开发者们青睐的架构模式之一。本文将详细介绍Android MVVM框架的基本概念、优势、实现流程以及一个实际案例。