终端研发部

kotlin开发经验谈1

2017-08-06  本文已影响798人  xingstarx

笔者最近几个月在学习kotlin,然后用来重构项目,反正是不断踩坑中,踩的很开心😊,这个过程中碰到了一些问题,值得记录下来, 供大家参考。

如何定义一个非空变量

本次的问题是如何定义一个非空变量,有哪些实现手段呢?

在此之前,我先讲讲如何定义常量变量,为了方便学习kotlin的语法特性,推荐使用idea开发工具。

那么现在创见一个java工程,选择kotlin支持。
创建一个文件,内容如下

class User {
    var name:String? = null
    val id:String? = null
}

fun main(args: Array<String>) {
    println("Hello World")

}

在这里我们定义了一个model类User,User类有两个属性,一个是变量name可空,一个是常量id可空

image.png

我们想知道这个文件最终对应的java代码是什么,有什么办法呢,我只会反编译额。接着就编译这个App.kt文件,idea中选择app.kt文件,然后单击右键Recompile App.kt,就会编译出class文件,接着我们用JD_GUI反编译工具,去查看这两个class文件的内容

image.png image.png

这个就是最终的反编译出来的java代码,我们可以理解为kotlin编译后得到的就是这样的java代码。
我们关注的重点是User类,id是定义的一个常量,反编译出来的代码里面将它设置了为final类型,默认的访问级别也是private,这也是kotlin的设计理念(不知道说的对不对_),属性私有化,对外提供geter,seter。而变量name,就跟我们平常定义java属性的方式类似,private String name,到此,如何定义一个常量变量就讲完了,对应生成的java代码我们也看到了,接下来回到正题,如何定义一个非空变量,有哪些手段?

先看一段代码片段吧(来自https://github.com/leanote/leanote-android)

image.png image.png

我们拿这个NoteFragment来描述,我需要在onCreate方法中初始化这个mNoteFragment,但是我不想让kotlin的代码中,出现mNoteFragment?.setMode(mode)这样的代码?

image.png

不想各个地方都出现这样的代码,如果把mNoteFragment定义为非空的那就好了,说干就干,该如何修改呢?

不可能直接这样改
var mNoteFragment: NoteFragment = NoteFragment.newInstance(),这样有问题的,就不多说了。

考虑到kotlin的语言特性我想到的只有
lateinit var mNoteFragment: NoteFragment 这样在onCreate方法里面初始化是可以的。
var mNoteFragment: NoteFragment by Delegates.notNull<NoteFragment>()和这种委托的模式

我们先看这一段代码,lateinit方式的

class MainActivity {
    lateinit var mNoteFragment: String
    fun onCreate() {
       mNoteFragment = newInstance()
    }
    companion object {
        fun newInstance():String {
            return "NoteFragment"
        }
    }
}

这种情况下生成的java代码如下

image.png

lateinit var mNoteFragment: String 被翻译成了@NotNull public String mNoteFragment mNoteFragment被定义为了public访问级别

接着看看委托模式下的

import kotlin.properties.Delegates

class MainActivity {
    var mNoteFragment: String by Delegates.notNull<String>()
    fun onCreate() {
       mNoteFragment = newInstance()
    }
    companion object {
        fun newInstance():String {
            return "NoteFragment"
        }
    }
}

反编译的java代码比较复杂,暂时也没有深究,感觉要麻烦多了

image.png

我们定义的mNoteFragment属性不见了,反而多了一个mNoteFragment$delegate属性,而且还是在构造方法中做的初始化操作,以及一个$$deletegatedProperties在静态代码块初始化的不知道什么东西的东西,虽然没有了mNoteFragment,但是还是有对应的geter,setter方法,具体就不深究了,暂时功力有限。

总结

到此为止,讲完了。
定义一个非空属性,基本上常用的手段就是lateinit和委托模式。

上一篇下一篇

猜你喜欢

热点阅读