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!"

惰性线程安全模式

  • 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"
}

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!"

检查初始化

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

3. 两者对比

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

示例场景

by lazy 适用场景:

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

    lateinit 适用场景:

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

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


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

目录
相关文章
|
3月前
|
Android开发
Android面试之Activity启动流程简述
Android面试之Activity启动流程简述
99 6
|
3月前
|
Android开发
Android面试高频知识点(1) 图解Android事件分发机制
Android面试高频知识点(1) 图解Android事件分发机制
|
3月前
|
消息中间件 存储 Java
Android面试高频知识点(2) 详解Android消息处理机制(Handler)
Android面试高频知识点(2) 详解Android消息处理机制(Handler)
|
3月前
|
XML 前端开发 Android开发
Android面试高频知识点(3) 详解Android View的绘制流程
Android面试高频知识点(3) 详解Android View的绘制流程
Android面试高频知识点(3) 详解Android View的绘制流程
|
3月前
|
存储 前端开发 测试技术
Android kotlin MVVM 架构简单示例入门
Android kotlin MVVM 架构简单示例入门
47 1
|
3月前
|
消息中间件 Android开发 索引
Android面试高频知识点(4) 详解Activity的启动流程
Android面试高频知识点(4) 详解Activity的启动流程
34 3
|
3月前
|
Android开发
Android面试之Activity启动流程简述
Android面试之Activity启动流程简述
24 0
|
3月前
|
JSON 调度 数据库
Android面试之5个Kotlin深度面试题:协程、密封类和高阶函数
本文首发于公众号“AntDream”,欢迎微信搜索“AntDream”或扫描文章底部二维码关注,和我一起每天进步一点点。文章详细解析了Kotlin中的协程、扩展函数、高阶函数、密封类及`inline`和`reified`关键字在Android开发中的应用,帮助读者更好地理解和使用这些特性。
43 1
|
4月前
|
Android开发 开发者 Kotlin
告别AsyncTask:一招教你用Kotlin协程重构Android应用,流畅度飙升的秘密武器
【9月更文挑战第13天】随着Android应用复杂度的增加,有效管理异步任务成为关键。Kotlin协程提供了一种优雅的并发操作处理方式,使异步编程更简单直观。本文通过具体示例介绍如何使用Kotlin协程优化Android应用性能,包括网络数据加载和UI更新。首先需在`build.gradle`中添加coroutines依赖。接着,通过定义挂起函数执行网络请求,并在`ViewModel`中使用`viewModelScope`启动协程,结合`Dispatchers.Main`更新UI,避免内存泄漏。使用协程不仅简化代码,还提升了程序健壮性。
122 1
|
5月前
|
调度 Android开发 开发者
【颠覆传统!】Kotlin协程魔法:解锁Android应用极速体验,带你领略多线程优化的无限魅力!
【8月更文挑战第12天】多线程对现代Android应用至关重要,能显著提升性能与体验。本文探讨Kotlin中的高效多线程实践。首先,理解主线程(UI线程)的角色,避免阻塞它。Kotlin协程作为轻量级线程,简化异步编程。示例展示了如何使用`kotlinx.coroutines`库创建协程,执行后台任务而不影响UI。此外,通过协程与Retrofit结合,实现了网络数据的异步加载,并安全地更新UI。协程不仅提高代码可读性,还能确保程序高效运行,不阻塞主线程,是构建高性能Android应用的关键。
69 4