笔者最近几个月在学习kotlin,然后用来重构项目,反正是不断踩坑中,踩的很开心?,这个过程中碰到了一些问题,值得记录下来, 供大家参考。
如何定义一个非空变量
本次的问题是如何定义一个非空变量,有哪些实现手段呢?
在此之前,我先讲讲如何定义常量变量,为了方便学习kotlin的语法特性,推荐使用idea开发工具。
那么现在创见一个java工程,选择kotlin支持。 创建一个文件,内容如下
class User { var name:String? = null val id:String? = null}fun main(args: Array) { println("Hello World")}复制代码
在这里我们定义了一个model类User,User类有两个属性,一个是变量name可空,一个是常量id可空
我们想知道这个文件最终对应的java代码是什么,有什么办法呢,我只会反编译额。接着就编译这个App.kt文件,idea中选择app.kt文件,然后单击右键Recompile App.kt,就会编译出class文件,接着我们用JD_GUI反编译工具,去查看这两个class文件的内容
这个就是最终的反编译出来的java代码,我们可以理解为kotlin编译后得到的就是这样的java代码。 我们关注的重点是User类,id是定义的一个常量,反编译出来的代码里面将它设置了为final类型,默认的访问级别也是private,这也是kotlin的设计理念(不知道说的对不对^_^),属性私有化,对外提供geter,seter。而变量name,就跟我们平常定义java属性的方式类似,private String name
,到此,如何定义一个常量变量就讲完了,对应生成的java代码我们也看到了,接下来回到正题,如何定义一个非空变量,有哪些手段?
先看一段代码片段吧(来自https://github.com/leanote/leanote-android)
我们拿这个NoteFragment来描述,我需要在onCreate方法中初始化这个mNoteFragment,但是我不想让kotlin的代码中,出现mNoteFragment?.setMode(mode)
这样的代码?
不想各个地方都出现这样的代码,如果把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代码如下
lateinit var mNoteFragment: String
被翻译成了@NotNull public String mNoteFragment
mNoteFragment被定义为了public访问级别
接着看看委托模式下的
import kotlin.properties.Delegatesclass MainActivity { var mNoteFragment: String by Delegates.notNull() fun onCreate() { mNoteFragment = newInstance() } companion object { fun newInstance():String { return "NoteFragment" } }}复制代码
反编译的java代码比较复杂,暂时也没有深究,感觉要麻烦多了
我们定义的mNoteFragment属性不见了,反而多了一个mNoteFragment$delegate属性,而且还是在构造方法中做的初始化操作,以及一个$$deletegatedProperties在静态代码块初始化的不知道什么东西的东西,虽然没有了mNoteFragment,但是还是有对应的geter,setter方法,具体就不深究了,暂时功力有限。
总结
到此为止,讲完了。 定义一个非空属性,基本上常用的手段就是lateinit和委托模式。