Android - ButterKnife 写给一知半解的同学。

ButterKnife GitHub 2019 年已经更新到 10.1.0 版本了,经过几天的学习和网上搜索资料,发现之前版本的一些疑难杂症已经不存在或者有些改善。刚开始用是非常爽的,如果真的要应用在项目中,有如下建议:

  1. 一个 module 撸到底的项目,直接用吧,没啥坑,都挺好。
  2. 大项目组件化的项目,可以尝试。低版本据搜索有很多坑,我在 10.1.0 版本实验了一下,配置得当没有问题。但我这个组件化写得很简单,如果你的项目更加复杂,就要动手试试了,需要试两个地方:一个看能不能编译通过;另一个看运行期间绑定的是否正确。

配置步骤

这个步骤按照 github 页面上的说明设置就行,这里用列表记录一下步骤:

给 application 模块配置:

  1. 设置为 Java8,介个 10.1.0 船新版本得用 8 了。
android {
  ...
  // Butterknife requires Java 8.
  compileOptions {
    sourceCompatibility JavaVersion.VERSION_1_8
    targetCompatibility JavaVersion.VERSION_1_8
  }
}
  1. 添加依赖:库 + 注解处理
dependencies {
  implementation 'com.jakewharton:butterknife:10.1.0'
  annotationProcessor 'com.jakewharton:butterknife-compiler:10.1.0'
}

给 library 模块设置:

  1. 先按照 application 模块的方法设置一遍
  2. 在 project level 的 gradle 文件中添加 buildscript 依赖
buildscript {
  repositories {
    google()
    jcenter()
   }
  dependencies {
    classpath 'com.android.tools.build:gradle:3.4.0'
    classpath 'com.jakewharton:butterknife-gradle-plugin:10.1.0' //  这里
  }
}
  1. 在 library 模块的 gradle 文件中 apply,写在 android library 插件的下面
apply plugin: 'com.android.library'
apply plugin: 'com.jakewharton.butterknife' //  这里

基本使用

举个例子

@BindView(R.id.clock_view) //  找到资源
View mClockView;           //  找到变量

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_lib_main);
    ButterKnife.bind(this); //  绑定
}

ButterKnife 的主要目的是简化资源和代码之间的绑定,使用起来很简单,主要有以下(看起来像废话一样的)几个步骤:

  1. 找到资源:R.id.clock_view
    • 如果是 application 模块,通过 R.id.xxx, R.string.xxx 等找到资源。
    • 如果是 library 模块,需要使用 ButterKnife 生成的 R2 来替代 R。原因如下:

因为注解中的 ElementValue 值(写在括号内的 R.id.xxx)必须是常量,而 R 在 library 中的 id 值都是变量。

为什么注解中的 ElementValue 必须是常量?因为使用注解时产生的所有信息在编译期间必须确定下来,直接写入注解内部的数据结构中。即使是 @Retention(RetentionPolicy.RUNTIME) 修饰的注解,也不可能在运行时动态运行一段字节码来计算,这样会徒增复杂度而且没有什么好处。

在 application 模块中,R 类中的标识符都是 final 的,也就是常量,是在编译期就能确定值的。而在 library 模块中,R 类中的标识符不是 final 的,在编译期无法确定。library 中不用常量的理由是 R 中的各种 id 值在一个 app 内必须是互不相同的,如果 library 模块在编译期就将这些 id 确定为常量的话,那么必须要考虑所有编译模块,会降低编译速度;而且 library 模块是共享的,如果使用了固定的 id 值,分发给其他项目使用难免会产生冲突。具体分析可以参考这个官方页面:Non-constant Fields in Case Labels

  1. 找到变量:View mClockView;
    • 不能声明成 private,因为要通过 ButterKnife 生成的类访问,而 private 修饰的成员只有自己才能访问。ButterKnife 生成的类与绑定的类在同一个包内,直接什么都不写用包访问权限就可以了。
    • 一旦写好变量和修饰它的注解,ButterKnife 就可以生成绑定的代码了,注意只是生成了绑定的代码,如果不调用这个生成的代码,也是没有完成绑定的。绑定的代码很简单,就是将找到的资源和变量关联起来。
  2. 绑定:ButterKnife.bind(this);
    • 这个步骤就是调用 ButterKnife 生成的绑定代码。在调用绑定代码之前,应该设置好 layout 文件,以便能通过资源 id 找到资源;在调用绑定代码之后,成员变量才绑定到资源上,这时才能访问成员变量。

    • 根据绑定变量所在的类的类型,bind 方法还有几个版本,比如绑定 Fragment 中的变量要用 bind(this, view),完整代码:

      @Override
      public View onCreateView(LayoutInflater inflater, ViewGroup container,
                                 Bundle savedInstanceState) {
          View view = inflater.inflate(R.layout.fragment_first, container, false);
          unbinder = ButterKnife.bind(this, view);
          return view;
      }
      
    • bind 方法的参数 target 必须是绑定的成员所在的类的对象。

    • bind 方法会返回一个 Unbinder 对象,可以用来解除绑定。但几乎所有情况都不需要手动调用,只有在 Fragment 中绑定才需要在 onDestroyView() 中手动解除。

原理简要分析

说白了 ButterKnife 就是替我们写了一些重复度很高的代码,我们只用关注绑定关系本身,重复的绑定代码都由 ButterKnife 生成。

那么凭什么 ButterKnife 能替我们写代码呢,就要借助注解(Annotation)和注解处理工具(APT)了。

简单说一下注解,注解就是带 @ 符号开头的修饰类、方法、变量等等的一些看起来不像代码的东西。Java 语言本身有一些内置的注解,最常见的要数 @Override 了。可以把注解当做给 Java 中的类、方法、字段等语法元素添加属性,再通过各种工具进行处理,来达到一定的目的。对 ButterKnife 来说,目的就是将资源和变量关联起来,不用再手动调用绑定的代码。ButterKnife 利用注解给成员赋予了资源 id 的属性,再经过 ButterKnife 的处理就可以将两者关联在一起。

那么是怎么关联的呢?这就要提一下注解的两种主要使用形式:反射和生成代码。ButterKnife 早期的版本就是用的反射来实现绑定,后来发现效率不如生成代码的实现方式,于是就改成了生成代码的方式。

生成代码的过程使用了注解处理工具(APT),它是一个运行在构建流程中的一个工具,可以读取到注解和被注解元素的信息,再通过自定义的处理器(也就是 gradle 依赖中的annotationProcessor 'com.jakewharton:butterknife-compiler:10.1.0')输出 Java 源代码,并将其纳入构建过程。

接下来从代码角度分析原理:

  1. 对于绑定的成员所在的类,ButterKnife 会生成一个对应的 Unbinder 类型,也就是实现了 Unbinder 接口的一个类。
    • 别看它叫 Unbinder,实际上绑定和解绑都是使用这个对象,绑定用构造方法,解绑用 unbind() 方法。
    • 这个类与被绑定的类在同一个包下,因此可以访问到包访问权限的成员。
    • 这个类的名字是按照规则生成的,被绑定类的名字加一个固定后缀就是这个 Unbinder 类型的名字。
  2. ButterKnife.bind(target, view) 方法内部会根据传入的 target 类型,通过名称规则拼接出生成的 Unbinder 类型的类名称,然后使用反射调用构造方法,也就是执行了绑定的代码。再将这个 Unbinder 对象返回,这样就可以通过这个对象调用 unbind() 方法来解除绑定。

组件化的影响

先说明一下我这个简单的组件化是怎么做的:

  1. 在 gradle.properties 文件中定义了一个变量 isModule_libdemo1=true 用来设置作为 library 还是作为 application。
  2. 在 library 模块的 build.grale 中,通过 isModule_libdemo1 的值来使用不同 plugin。
if (isModule_libdemo1.toBoolean()) {
    apply plugin: 'com.android.library'
} else {
    apply plugin: 'com.android.application'
}
  1. 使用不同的 manifest 文件(略)

ButterKnife 在组件化过程中有什么坑吗?

由于是 library 模块,代码用的都是 R2 对象,一旦切换 plugin,由 library 变成 application,就没有 R2 对象了吗,就应该用 R 对象了吗?

if (isModule_libdemo1.toBoolean()) {
    apply plugin: 'com.android.library'
} else {
    apply plugin: 'com.android.application'
}
apply plugin: 'com.jakewharton.butterknife'

即使是 application,也可以使用 ButterKnife 插件,可以生成 R2 对象,因此应该统一使用 R2 来寻找资源。

其他注意事项

  • 重构资源名称:用重构工具可以帮你快速修改。但 ButterKnife 生成的 R2 中的名字 Android Studio 可不管,这时编译一下就能发现问题。但最好不要改成原来就有的名字,会没有任何提示默默编译成功,在运行的时候给你出错。
  • @OnClick 与 R2:在 @OnClick(id) 修饰的方法中,即使 id 是 R2 的写法,方法体内部仍要使用 R 对象中的 id 来区分多个按钮的 id。可以这么理解,R2 是 ButterKnife 为了绕开资源 id 非常量的问题,R2R 是能一一对应的,因此能够在生成的 Unbinder 代码使用正确的 R 中的 id。

(ole)

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,039评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,223评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,916评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,009评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,030评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,011评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,934评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,754评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,202评论 1 309
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,433评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,590评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,321评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,917评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,568评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,738评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,583评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,482评论 2 352

推荐阅读更多精彩内容