前言
(PS:因为Android Stuidio 3.0后,会自动导入插件,所以文章只针对之前版本或其他idea)
另一个Kotlin团队研发的可以让开发更简单的插件是 Kotlin Android
Extensions 。当前仅仅包括了view的绑定。这个插件自动创建了很多的属性来让
我们直接访问XML中的view。这种方式不需要你在开始使用之前明确地从布局中去
找到这些views。
这些属性的名字就是来自对应view的id,所以我们取id的时候要十分小心,因为它
们将会是我们类中非常重要的一部分。这些属性的类型也是来自XML中的,所以我
们不需要去进行额外的类型转换。
Kotlin Android Extensions 的一个优点是它不需要在我们的代码中依赖其它
额外的库。它仅仅由插件组层,需要时用于生成工作所需的代码,只需要依赖于
Kotlin的标准库。
那它背后是怎么工作的?该插件会代替任何属性调用函数,比如获取到view并具有
缓存功能,以免每次属性被调用都会去重新获取这个view。需要注意的是这个缓存
装置只会在 Activity 或者 Fragment 中才有效。如果它是在一个扩展函数中增
加的,这个缓存就会被跳过,因为它可以被用在 Activity 中但是插件不能被修
改,所以不需要再去增加一个缓存功能。
怎么去使用Kotlin Android Extensions
如果你还记得,现在项目已经准备好去使用Kotlin Android Extensions。当我们创建
这个项目,我们就已经在 build.gradle 中增加了这个依赖:
buldscript{
repositories {
jcenter()
}
dependencies {
classpath "org.jetbrains.kotlin:kotlin-android-extensions:$kotlin_version"
}
}
唯一一件需要这个插件做的事情是在类中增加一个特定的"手工" import 来使用这
个功能。我们有两个方法来使用它:
Activities 或者 Fragments 的 Android Extensions
这是最典型的使用方式。它们可以作为 activity 或 fragment 的属性是可以被
访问的。属性的名字就是XML中对应view的id。
我们需要使用的 import 语句以 kotlin.android.synthetic 开头,然后加上我
们要绑定到Activity的布局XML的名字:
import kotlinx.android.synthetic.activity_main.*
此后,我们就可以在 setContentView 被调用后访问这些view。新的Android
Studio版本中可以通过使用 include 标签在Activity默认布局中增加内嵌的布局。
很重要的一点是,针对这些布局,我们也需要增加手工的import:
import kotlinx.android.synthetic.activity_main.*
import kotlinx.android.synthetic.content_main.*
Views 的 Android Extensions
前面说的使用还是有局限性的,因为可能有很多代码需要访问XML中的view。比
如,一个自定义view或者一个adapter。举个例子,绑定一个xml中的view到另一个
view。唯一不同的就是需要 import :
import kotlinx.android.synthetic.view_item.view.*
如果我们需要一个adapter,比如,我们现在要从inflater的View中访问属性:
view.textView.text = "Hello"
重构我们的代码
现在是时候使用 Kotlin Android Extensions 来修改我们的代码了。修改相当
简单。
我们从 MainActivity 开始。我们当前只是使用了 forecastList 的
RecyclerView。但是我们可以简化一点代码。首先,为 activity_main XML增加
手工import:
import kotlinx.android.synthetic.activity_main.*
之前说过,我们使用id来访问views。所以我要修改 RecyclerView 的id,不使用
下划线,让它更加适合Kotlin变量的名字。XML最后如下:
<FrameLayout
xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent">
<android.support.v7.widget.RecyclerView
android:id="@+id/forecastList"
android:layout_width="match_parent"
android:layout_height="match_parent"/>
</FrameLayout>
然后现在,我们可以不需要 find 这一行了:
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_main)
forecastList.layoutManager = LinearLayoutManager(this)
...
}
这已经是最小的简化了,因为这个布局非常简单。但是 ForecastListAdapter 也
可以从这个插件中受益。这里你可以使用一个装置来绑定这些属性到view中,它可
以帮助我们移除所有 ViewHolder 的 find 代码。
首先,为 item_forecast 增加手工导入:
import kotlinx.android.synthetic.item_forecast.view.*
然后现在我们可以在 ViewHolder 中使用包含在 itemView 中的属性。实际上你
可以在任何view中使用这些属性,但是很显然如果view不包含要获取的子view就会
奔溃。
现在我们可以直接访问view的属性了:
class ViewHolder(view: View, val itemClick: (Forecast) ->Unit):RecyclerView.ViewHolder(view) {
fun bindForecast(forecast: Forecast) {
with(forecast){
Picasso.with(itemView.ctx).load(iconUrl).into(itemView.icon)
itemView.date.text = date
itemView.description.text = description
itemView.maxTemperature.text = "${high.toString()}"
itemView.minTemperature.text = "${low.toString()}"
itemView.onClick { itemClick(forecast) }
}
}
}
Kotlin Android Extensions插件帮助我们减少了很多模版代码,并且简化了我们访问
view的方式。