Android经典面试题之Kotlin延迟初始化的by lazy和lateinit有什么区别?

简介: **Kotlin中的`by lazy`和`lateinit`都是延迟初始化技术。`by lazy`用于只读属性,线程安全,首次访问时初始化;`lateinit`用于可变属性,需手动初始化,非线程安全。`by lazy`支持线程安全模式选择,而`lateinit`适用于构造函数后初始化。选择依赖于属性特性和使用场景。**

本文首发于公众号“AntDream”,欢迎微信搜索“AntDream”或扫描文章底部二维码关注,和我一起每天进步一点点

在Kotlin中,by lazylateinit 都是用于延迟初始化的手段,但它们各自有不同的用法和特点。下面详细讨论它们的作用和区别。

1. by lazy

作用

by lazy 是一种委托属性,用于延迟初始化一个只读属性。属性在第一次访问时才会被初始化,并且初始化操作只会执行一次。

用法

语法:val property: Type by lazy { initializer }

lazy 的默认线程安全模式是 LazyThreadSafetyMode.SYNCHRONIZED,它确保多线程环境下属性只会被初始化一次。

示例

val myValue: String by lazy {
    println("Computed only once")
    "Hello, World!"
}

// 第一次访问 myValue,会触发初始化代码块执行
println(myValue) // 输出: "Computed only once" 和 "Hello, World!"
// 之后的访问不会重复执行初始化代码块
println(myValue) // 输出: "Hello, World!"
AI 代码解读

惰性线程安全模式

  • LazyThreadSafetyMode.SYNCHRONIZED:默认值,确保多线程环境下属性只能被初始化一次。
  • LazyThreadSafetyMode.PUBLICATION:允许多个线程在同一时间初始化,但只使用第一个完成的结果。
  • LazyThreadSafetyMode.NONE:不进行任何同步,适用于单线程环境。

示例

val valueSynchronized: String by lazy(LazyThreadSafetyMode.SYNCHRONIZED) {
    "Synchronized"
}

val valuePublication: String by lazy(LazyThreadSafetyMode.PUBLICATION) {
    "Publication"
}

val valueNone: String by lazy(LazyThreadSafetyMode.NONE) {
    "None"
}
AI 代码解读

2. lateinit

作用

lateinit 是一种延迟初始化的关键字,用于延迟初始化一个 var 可变属性。属性类型必须是非空的且不能是原始类型(如 Int, Double)。

用法

语法:lateinit var property: Type

lateinit 属性不能有自定义的 getter 和 setter,必须在使用之前显式初始化,否则会抛出 UninitializedPropertyAccessException

示例

lateinit var myValue: String

fun initialize() {
    myValue = "Hello, World!"
}

// 使用之前必须显式初始化
initialize()
println(myValue) // 输出: "Hello, World!"
AI 代码解读

检查初始化

  • 可以使用 ::property.isInitialized 语法来检查属性是否已经初始化。
    if (::myValue.isInitialized) {
      println(myValue)
    } else {
      println("myValue is not initialized")
    }
    
    AI 代码解读

3. 两者对比

特性 by lazy lateinit
适用类型 val(只读属性) var(可变属性)
初始化时机 第一次访问时 必须手动初始化
线程安全 默认线程安全(可选择不同的线程安全模式) 非线程安全
Nullability 支持不可空类型 支持不可空类型(不能用于原始类型)
属性检查 不需要显式检查 可以通过 ::property.isInitialized 检查
自定义 getter/setter 不支持 不支持
使用场景 用于只读且惰性初始化的属性 用于需要在构造函数之外初始化的可变属性

示例场景

by lazy 适用场景:

  • 需要惰性初始化不可变的属性。
  • 需要线程安全的初始化或者只在单线程中操作。

    lateinit 适用场景:

    • 需要在构造方法之后初始化的可变属性。
  • 需要在某个特定操作时才对属性进行赋值。

总结来说,选择使用 by lazy 还是 lateinit 要依据属性的特性和具体的使用场景。by lazy 更适合不可变的延迟初始化场合,而 lateinit 则适用于在构造方法之后需要手动初始化的可变属性。


欢迎关注我的公众号AntDream查看更多精彩文章!

目录
打赏
0
5
5
0
148
分享
相关文章
Android实战经验之Kotlin中快速实现MVI架构
MVI架构通过单向数据流和不可变状态,提供了一种清晰、可预测的状态管理方式。在Kotlin中实现MVI架构,不仅提高了代码的可维护性和可测试性,还能更好地应对复杂的UI交互和状态管理。通过本文的介绍,希望开发者能够掌握MVI架构的核心思想,并在实际项目中灵活应用。
60 8
Android经典面试题之Kotlin中Lambda表达式和匿名函数的区别
Lambda表达式和匿名函数都是Kotlin中强大的特性,帮助开发者编写简洁而高效的代码。理解它们的区别和适用场景,有助于选择最合适的方式来解决问题。希望本文的详细讲解和示例能够帮助你在Kotlin开发中更好地运用这些特性。
51 9
Android面试之Activity启动流程简述
Android面试之Activity启动流程简述
137 6
Android面试高频知识点(1) 图解Android事件分发机制
Android面试高频知识点(1) 图解Android事件分发机制
Android面试高频知识点(2) 详解Android消息处理机制(Handler)
Android面试高频知识点(2) 详解Android消息处理机制(Handler)
Android面试高频知识点(4) 详解Activity的启动流程
Android面试高频知识点(4) 详解Activity的启动流程
53 3
Android kotlin MVVM 架构简单示例入门
Android kotlin MVVM 架构简单示例入门
75 1
Android面试高频知识点(3) 详解Android View的绘制流程
Android面试高频知识点(3) 详解Android View的绘制流程
49 2
Android面试之Activity启动流程简述
Android面试之Activity启动流程简述
44 0

热门文章

最新文章