前言
该博客是《第三行代码》第三版第4章Kotlin部分的笔记,方便记忆和复习
Kotlin延迟初始化和密封类
一、对变量延迟初始化
如果类中存在很多全局变量实例,为了保证它们能够满足Kotlin的空指针检查语法标准,就必须做许多的非空判断才行,即使我们可以确保它不为空。
class Test {
private var adapter: MsgAdapter? = null
override fun onCreate() {
···
adapter = MsgAdapter(msgList)
····
}
override fun onClick() {
···
adapter?.notify();
···
}
}
延迟初始化使用的是lateinit关键字,它可以告诉Kotlin编译器,我们会在晚些时候对这个变量进行初始化,这样就不用在一开始的时候将它赋值为null了
class Test {
private lateinit var adapter: MsgAdapter
override fun onCreate() {
···
adapter = MsgAdapter(msgList)
····
}
override fun onClick() {
···
adapter.notify();
···
}
}
我们在adapter变量前面加上lateinit关键字,这样就不用一开始就将它赋值为null了,同时类型声明也就可以改成MsgAdapter了,由于MsgAdapter是不可空类型,这样子在onClick中就可以不用进行判空处理了。
使用lateinit并不是没有风险的,如果我们adapter还没有初始化的情况下直接使用它,那么程序就一定会崩溃,并且抛出一个UninitializedPropertyAccessException异常
我们可以通过代码来判断一个全局变量是否已经完成了初始化,这样子就能避免一个变量重复进行初始化。
class Test {
private lateinit var adapter: MsgAdapter
override fun onCreate() {
···
if(!::adapter.isInitialized) {
adapter = MsgAdapter(msgList)
}
····
}
override fun onClick() {
···
adapter.notify();
···
}
}
::adapter.isInitialized可用于判断adapter是否已经初始化
二、密封类
定义如下一段代码:
interface Result
class Success(val msg: String) : Result
class Failure(val error: Exception) : Result
再定义一个getResultMsg()方法
fun getResultMsg(result: Result) = when (result) {
is Success -> result.msg
is Failure -> result.error.message
else -> throw IllegalArgumentException()
}
由于语法的限制,我们不得不编写一个else语句,尽管这个else语句永远不可能运行。
可以使用密封类解决该问题
密封类的关键字是sealed class,我们可以将Result接口改造成密封类
sealed class Result
class Success(val msg: String) : Result()
class Failure(val error: Exception) : Result()
这样子,getResultMsg()方法中的else条件已经不再需要了。
fun getResultMsg(result: Result) = when (result) {
is Success -> result.msg
is Failure -> result.error.message
}