在 Kotlin 中使用 Lazy 和 Lateinit 初始化属性
本文介绍了帮助初始化属性的 lazy()
和 lateinit
的 Kotlin。我们将从了解这些属性初始化器的作用开始,然后了解两者之间的区别。
如果我们想在构造函数之外或类体底部的某个地方初始化一个属性,我们只有两个选择:延迟初始化或 lateinit
。
在 Kotlin 中使用 by lazy
初始化属性
延迟初始化是在 lazy()
函数的帮助下完成的。它接受一个 lambda 作为输入并返回一个 lazy<t>
的实例。
该实例用于实现惰性属性所需的委托。默认情况下,惰性属性的评估是同步的。
同步意味着尽管编译器在单个线程中计算值,但所有线程将看到相同的值。
Kotlin 允许使用 LazyThreadSafetyMode.PUBLICATION
作为 lazy()
函数的参数来防止同步。
例子:
val lazyVal: String by lazy {
println("This is an")
"example of lazy initialization!"
}
fun main() {
println(lazyVal)
}
输出:
在 Kotlin 中使用 lateinit
初始化属性
后期初始化允许在构造函数之外初始化非空类型属性,同时避免空检查。
通常,非空属性在构造函数中初始化。但是在某些情况下,我们可能希望不这样做,例如设置单元测试方法。
lateinit
修饰符有助于解决此类问题。我们可以将此初始化程序与 var
或可变属性一起使用。
关于 lateinit
修饰符需要记住的几点是:
- 我们可以在主构造函数之外使用
lateinit
- 我们只能将
lateinit
用于没有自定义getters()
和setters()
的属性。 - 属性的类型应该是非空的和原始的
例子:
lateinit var student1:Student
fun main(args: Array<String>) {
student1 = Student("David Henry's",303)
print(student1.Name + " ID is " + student1.ID.toString())
}
data class Student(var Name:String, var ID:Int);
输出:
Kotlin 中 by lazy
和 lateinit
的区别
by lazy
和 lateinit
修饰符之间存在几个显着差异。这里是其中的一些:
by lazy
用于初始化val
属性,lateinit
用于初始化var
属性。lateinit
使用支持字段来存储结果的值,by lazy
使用委托对象。- 我们不能将
lateinit
与可空属性或 Java 原始类型一起使用。 - 我们可以从编译器可以看到对象的任何地方初始化
lateinit
并支持多个初始化。另一方面,by lazy
只允许进行一次初始化,该初始化只能通过覆盖子类中的属性来更改。 by lazy
初始化默认是线程安全的,lateinit var
修饰符跨多个线程的初始化取决于用户的代码。- 我们可以为多个属性保存、传递甚至使用
by lazy
初始化程序的实例。但是lateinit
中没有额外的运行时状态。
在 Kotlin 中何时使用 by lazy
我们可以使用 by lazy
时,
- 我们不希望变量初始化,除非我们调用或使用它们;
- 我们想初始化一个变量,然后在整个代码中使用它;和
- 我们正在使用只读属性。
在 Kotlin 中何时使用 lateinit
我们可以在以下情况下使用 lateinit
,
- 我们要延迟初始化一个变量;
- 我们想在使用它之前初始化一个变量;
- 我们正在使用
var
或可变属性;和 - 变量可能会在稍后阶段发生变化。
Kailash Vaviya is a freelance writer who started writing in 2019 and has never stopped since then as he fell in love with it. He has a soft corner for technology and likes to read, learn, and write about it. His content is focused on providing information to help build a brand presence and gain engagement.
LinkedIn