kotlin杂记之延迟初始化和密封类

前言

该博客是《第三行代码》第三版第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
}
Kotlin 中,单例初始化参数可以通过伴生对象的方式传入。具体来说,可以在单例内部定义一个带有参数的实例化方法,这个方法会取代的构造方法,然后在伴生对象中定义一个名为 INSTANCE 的静态变量,用于保存单例对象的实例。在使用单例对象的时候,可以直接通过访问 INSTANCE 变量来调用方法。举个例子,下面是一个带有初始化参数的单例的示例代码: ``` class MySingleton private constructor(val name: String) { fun sayHello() { println("Hello, I'm $name.") } companion object { private var instance: MySingleton? = null fun getInstance(name: String): MySingleton { instance = instance ?: MySingleton(name) return instance!! } } } fun main() { val s1 = MySingleton.getInstance("John") val s2 = MySingleton.getInstance("Mary") s1.sayHello() // 输出:Hello, I'm John. s2.sayHello() // 输出:Hello, I'm John. } ``` 在上面的示例中,我们定义了一个名为 MySingleton 的单例,并在构造方法中传入一个名为 name 的字符串参数。为了避免外界直接通过构造方法来创建单例对象,我们将其私有化,并在内部定义了一个名为 getInstance 的实例化方法,在这个方法中会判断单例对象是否已经创建,如果没有,则创建一个新对象并保存到 instance 变量中,最后返回该对象。在 main 函数中我们分别通过 getInstance 方法获取了两个 MySingleton 对象,并调用它的 sayHello 方法来输出两个对象的名字。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值