从java转kotlin开发APP(一)

#初步使用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替换为当前布局就可以
image.png

使用控件时直接使用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) //使用正常

仍有未解开之谜,文档继续更新 未完待续。。。

猜你喜欢

转载自my.oschina.net/u/3491516/blog/1788950