从 Butter Knife 到 Kotter Knife 再到 Kotlin Android Extensions

Butter Knife

Butter Knife 是安卓开发中常用的一种 View 绑定框架,主要用来减少 View 的获取&强转的样板代码。

原生的安卓 Java 代码中,控件需要自己手动获取和强制转换。

ListView simpleListView;
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
    // ...
    View view = findViewById(R.id.simple_list);
    simpleListView = (ListView) view;
    simpleListView.setAdapter(adapter);
}

如上面代码所示,至少要经历三个步骤:

  1. 声明 View 变量,包含 View 的具体类型(simpleListView 对象,ListView 类型)
  2. 调用 findViewById 方法获取资源的 View 对象
  3. 将 View 对象强制转换成对应类型的 VIew(ListView)

虽然 2-3 步可以简化成一行代码,但是经历的步骤一定是分明的。

但是通过 View 绑定,这个步骤可以简化到一步:


@BindView(R.id.simple_list) ListView simpleListView;
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
    // ...
    simpleListView.setAdapter(adapter);
}

没有 simpleListView 的赋值过程,当然更不会有类型转换代码。因为这之间的步骤 Butter Knife 框架帮你做了。你只用告诉它资源 ID 和接收对象即可,也就是 BindView 注解的作用。

Kotlin 是怎么做的

var simpleListView: ListView? = null
simpleListView = findViewById(R.id.simple_list) as ListView
simpleListView?.adapter = adapter

在 Kotlin 中也逃不过这三个步骤,但是相比 Java 的优点是强制转换只需要 as 关键字,手敲几乎没不方便的地方。用 Java 的情况下,大家几乎都是在等号右边直接调用 findViewById 然后按 <Ctrl + Enter> 快捷键让 IDE 自动纠正代码这种方式。。。

所以面对这种问题,即便是语法灵活得多的 Kotlin 也没有太大优势,类型的强制转换总是非常令人厌恶的。而且 Kotlin 原生并不支持 Butter Knife 。

让 Kotlin 也用上 View 绑定

让 Kotlin 用上方便的 View 绑定功能主要是两种方式,它们都挺简单:

  1. 让 Butter Knife 在 Kotlin 上正常工作
  2. 选择原生 Kotlin 所支持的 View 绑定框架

第一种方式:

@BindView(R.id.simple_list) @JvmField var simpleListView: ListView? = null

JvmField 注解让 Kotlin 实例的字段具有与底层相同的可见性,即对 Java 是可见的,当然它的前提必须是非私有属性。这点对于需要属性注入的情况是必须的,同时也是 Kotter Knife 原生不能支持 Kotlin 的原因。

例如 Kotlin 中并没有 “静态变量” 这个元素,但是提供了 companion object 来模拟静态的调用方式。但是 companion object 在底层仍然不是静态的,这对于 Java 而言企图通过静态调用 Kotlin 的 companion object 里边的内容是不行的。
想让 Kotlin 在底层产生静态实例,需要这样做:

class Key(val value: Int) {
    companion object {
        @JvmField
        val COMPARATOR: Comparator<Key> = compareBy<Key> { it.value }
    }
}

此时的 Key.COMPARATOR 对于 Java(或者其它 JVM 语言)都是是静态的,它们在底层采样同样的方式储存。第一种方式之所以能解决也是类似的道理。

第二种方式:

Kotter Knife 是为 Kotlin 语言所写的 View 绑定框架,它这样来使用:

val impleListView: ListView by bindView(R.id.simple_list)

从形式上来看,就是把给属性加注解换成了对属性访问进行委托,委托给 bind* 函数。如果你不了解什么是委托,请看这里
注意:Kotlin 中的 by 关键字只是省略了委托中的样板代码,和委托设计模式的思想是一模一样的。此处的属性委托背后的实现也非常简单,属性会被延迟计算,第一次访问时进行 View 的查找和转换,如果没有找到则会抛出异常,异常实例是:

IllegalStateException("View ID $id for '${desc.name}' not found.")

究竟应不应该因为一个注解问题放弃 Butter Knife?
首先你要明白,毕竟 @JvmField 也是 Kotlin 语言重要部分的元素之一,这个重要的部分就是:和 Java 的交互调用。所以 Kotlin 并不算是不支持 Butter Knife,在需要它的时候不用多虑,毫无疑问可以当做完全兼容的 Java 类库使用。

Kotlin Android Extensions

然而说到这里,本文的主角还未介绍过... 因为介绍它的时候就是抛弃上述所有东西的时候。你可以将它当做 Kotlin 官方对安卓开发提供的加强支持:它包括了 View 绑定,并且是一种更方便的新形式。它就是: Kotlin Android Extensions

给项目模块的 build.gradle 添加配置:

apply plugin: 'kotlin-android-extensions'

buildscript {
    repositories {
        jcenter()
    }
    dependencies {
        classpath "org.jetbrains.kotlin:kotlin-android-extensions:$kotlin_version"
    }
}

然后,就可以直接使用生成的对象了:

// 不可少的 import
import kotlinx.android.synthetic.main.fragment_main.*
// 省略其它 import
class MyFragment : Fragment {
    override fun onViewCreated(view: View?, savedInstanceState: Bundle?) {
        super.onViewCreated(view, savedInstanceState)
        // 直接使用 ListView 对象
        simple_list.adapter = adapter
    }
    // ...
}

注意:我没有声明任何的 ListView 对象,更没有相应的赋值操作,simple_list 是资源 ID。也就是说:我直接将资源 ID 名 simple_list 当做该 ListView 的对象实例使用。

当然,上面的 import 是不能少的,import 的规则是:

import kotlinx.android.synthetic.main.<layout>.*

即 import 了相应的 layout ,就能直接使用里边具有 id 属性的 View 实例,将 1-2-3 个步骤全部省略,可谓是最方便的形式。

最后

虽然 Butter Knife 非常优秀,但是既然我能更优雅的解决问题,还能减少依赖,何乐不为。所以:既然你用上了 Kotlin,那么请丢弃所有的 View 注入框架。

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

推荐阅读更多精彩内容