初步使用Kotlin编写代码,并替换项目中的Butterknife
(记-kotlin配置,Butterknife替换,View层使用问题)
1.开发工具 android studio-2.3.3
2.在项目根目录下的build.gradle文件添加一下代码
buildscript {
ext.kotlin_version = '1.2.21'
dependencies {
classpath 'com.android.tools.build:gradle:2.3.3'
classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlin_version"
}
}
再在app模块的build.gradle中添加以下代码
apply plugin: 'kotlin-android'
apply plugin: 'kotlin-android-extensions'
以上完成了在android studio3.0以下写kotlin的准备操作(android studio3.0说是自带了,未升级不以确定)
3.替代项目中的Butterknife库。
在app模块的build.gradle文件中添加的插件就是用于替代Butterknife的。
apply plugin: 'kotlin-android-extensions'
不需要findById,不需要BindView,在项目中可以直接使用View的id来进行操作View,原理如下(取自网络百度可查)
Kotlin Android Extensions是另一个Kotlin团队研发的插件,让我们用更少的代码来开发程序 。 当前仅仅包括了view的绑定。该插件自动创建了很多的属性来让我们直接访问XML中的view。因此不需要你在布局中去找到这些views。 我们使用的View,其名字就是来自对应view的id,所以我们取id的时候要十分小心,这将会是我们类中非常重要的一部分。 这些控件的类型也是来自XML中的,所以我们不需要去进行额外的类型转换。Kotlin Android Extensions使用不需要依赖其它额外的库。它仅仅由插件组层,用于生成工作所需的代码,只需依赖于Kotlin的标准库。 Kotlin Android Extensions工作原理是: 该插件会代替任何属性调用函数,比如获取到view并具有缓存功能,以免每次属性被调用都会去重新获取这个view。 这个缓存装置只会在Activity或者Fragment中才有效。如果它是在一个扩展函数中增加的
代码操作
在需要使用的Activity/Fragment类中,需要设置插件的布局来源,即下代码
import kotlinx.android.synthetic.main.activity_repository.*
其中 kotlinx.android.synthetic.main.是固定路径,会有自动提示补全的,是不需要改动的,只需要将后面的activity_repository替换为当前布局就可以
使用控件时直接使用View的id再引用属性,比如设置View的visibility属性和点击事件
Xml
<TextView
android:id="@+id/repository_name"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginStart="@dimen/activity_horizontal_margin"
android:layout_marginTop="@dimen/activity_horizontal_margin"
android:textSize="20sp"
tools:text="@string/name"/>
kotlin
// layoutContent 是id
repository_name.visibility = View.GONE
// 设置点击事件
repository_name.setOnClickListener {
Toast.makeText(this, "这是个点击事件", Toast.LENGTH_LONG).show()
}
java
// 原生方法
// repositoryName是对象名称
repositoryName = (TextView)layout.findViewById(R.id.repository_name);
repositoryName.setVisibility(View.GONE);
checkEvent.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Toast.makeText(this, "这是个点击事件", Toast.LENGTH_LONG).show();
}
});
// Butterknife使用
@BindView(R.id.repository_name)
TextView repositoryName;
@OnClick(R.id.repository_name)
public void nameOnclick(){
Toast.makeText(this, "这是个点击事件", Toast.LENGTH_LONG).show();
}
在添加kotlin插件之后,使用View直接调用id就可以使用该控件的属性了,常用的setText(),setVisibility()等方法直接使用 .text和.visibility
单从常用的属性设置和点击事件来看就可以减少非常多的代码量了。
kotlin还有一个便捷的语法细节,那就是每一行代码后面不需要“;” 每一行代码就一个语句,从语法上根治代码的可阅读性。
注意,无法在java代码中写kotlin代码,kotlin代码中写java代码,两者可以互相调用。
kotlin和java是两种语言,这个是可以理解的。
问题
问题1:
在kotlin-android-extensions 插件版本为1.1.4-3时 ,是不支持XML使用include标签的,在1.2.21版本时是可以使用的,但是在使用include标签的时候不要给include命名ID,不然会出现控件findById失败的错误,调用的控件返回NULL,虽然后面再调用kotlin自带方法又是返回正常的
var i = layout.findViewById<TextView>(R.id.tv_title)
问题2:
在Fragment中使用时有需要注意的,当我们使用Butterknife时,在调用bind之后就可以对View操作了。
View layout = LayoutInflater.from(getActivity()).inflate(R.layout.activity_repository, null);
ButterKnife.bind(this, layout);
而kotlin-android-extensions插件在onCreateView方法return view结束之前使用view是会抛出NULL异常的,原因是此时并没有缓存Layout,只有在执行完onCreateView方法之后才会缓存Layout。
那么许多控件的初始化就需要换一个方法内调用了。
按fragment的生命周期,我们可以在
onActivityCreated
onViewCreated
这两个方法内初始化VIew
而activity不存在这个问题,跟以前的使用一样的,执行完setContentView方法就可以直接调用ID操作View了。
setContentView(R.layout.activity_organization)
问题3:
前面提到控件的属性使用,在EditText中设置框内文本按照通用设置逻辑
name_value.text = datdBean.consigneeName // 使用报错
上面的代码使用时报错,错误日志是type is String! but Editable! was expected,很明显类型不对,很尴尬,这个地方用 .属性方式赋值是不行的,需要直接调用未封装过的API
name_value.setText(datdBean.consigneeName) //使用正常
仍有未解开之谜,文档继续更新
未完待续。。。