首页 最新 热门 推荐

  • 首页
  • 最新
  • 热门
  • 推荐

一文带你吃透Kotlin中 lateinit 和 by lazy 的区别和用法

  • 25-04-16 14:00
  • 3552
  • 8873
juejin.cn

在 Kotlin 中,lateinit 和 by lazy 都是用于处理延迟初始化的机制,但它们的实现方式、适用场景和特性有显著差异。本文将从原理、用法到实际场景结合示例代码的进行讲解。

一、 lateinit 的特点与使用场景

1、特性

  • 修饰可变变量:仅用于 var 声明。
  • 手动初始化:开发者需在适当位置(如生命周期回调)显式初始化。
  • 非空类型:只能用于非空类型(如 String、View),不支持基本数据类型(如 Int、Boolean)。
  • 异常风险:访问未初始化的变量会抛出 UninitializedPropertyAccessException。
  • 无线程安全:需自行处理多线程环境下的初始化。

2、适用场景

  • Android 组件初始化:如 Activity/Fragment 中的 View 绑定。
  • 依赖注入:框架(如 Dagger)在运行时注入的变量。
  • 明确生命周期:确保在使用前完成初始化(如 onCreate() 中初始化)。

3、示例代码

kotlin
代码解读
复制代码
class MyActivity : AppCompatActivity() { private lateinit var button: Button // 非空,延迟初始化 override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) setContentView(R.layout.activity_main) button = findViewById(R.id.btn_submit) // 手动初始化 button.setOnClickListener { /* ... */ } } }

二、 by lazy 的特点与使用场景

1、特性

  • 修饰只读变量:仅用于 val 声明。
  • 自动初始化:首次访问时执行 Lambda 表达式并缓存结果。
  • 支持所有类型:包括基本数据类型(如 Int、Boolean)。
  • 线程安全:默认使用 LazyThreadSafetyMode.SYNCHRONIZED(安全但略慢),可自定义模式。
  • 无异常风险:首次访问时必然初始化。

2、适用场景

  • 高开销初始化:如数据库连接、文件读取。
  • 单例模式:确保全局唯一实例。
  • 条件性初始化:仅在需要时才创建对象。

3、示例代码

kotlin
代码解读
复制代码
class MyService { // 高开销资源,首次访问时初始化 private val database: Database by lazy { Database.connect("jdbc:mysql://localhost:3306/mydb") } fun queryData() { val result = database.query("SELECT * FROM users") // 首次调用时初始化 // ... } }

三、核心对比表格

特性lateinit varval by lazy
变量类型可变 (var)只读 (val)
初始化时机手动显式初始化首次访问时自动初始化
适用数据类型非空对象类型(不支持基本类型)所有类型(包括基本类型)
线程安全需自行处理默认线程安全(可配置模式)
异常风险未初始化时抛出异常无(确保首次访问时初始化)
典型场景Android View 绑定、依赖注入单例、高开销资源、延迟计算

四、如何选择?

1、选择 lateinit 当:

  • 变量需要重新赋值(var)。
  • 初始化时机明确(如生命周期方法中)。
  • 处理非空对象且无法使用 by lazy(如基本类型不适用)。

2、选择 by lazy 当:

  • 变量只需初始化一次且不可变(val)。
  • 需要延迟初始化直到首次使用,减少启动开销。
  • 需要线程安全的延迟初始化。

五、高级用法与注意事项

1、by lazy 的线程模式

kotlin
代码解读
复制代码
val data: List by lazy(LazyThreadSafetyMode.NONE) { // 非线程安全模式,适用于单线程环境 loadExpensiveData() }

2、lateinit 的初始化检查

kotlin
代码解读
复制代码
if (::button.isInitialized) { // 使用反射检查是否初始化 button.text = "Click Me" }

3、避免陷阱

  • lateinit 未初始化:确保在使用前初始化,或通过 isInitialized 检查。
  • by lazy 的副作用:Lambda 中的代码应幂等,避免重复执行产生意外结果。

六、总结

  • lateinit:适用于可变、生命周期明确的对象,需手动控制初始化。
  • by lazy:适用于只读、高开销或按需初始化的资源,自动处理线程安全。

根据变量是否需要可变、初始化成本及线程需求,合理选择二者以提升代码效率和安全性。

更多分享

  1. 一文吃透Kotlin中冷流(Clod Flow)和热流(Hot Flow)
  2. 一文带你吃透Kotlin协程的launch()和async()的区别
  3. 一文带你吃透接口(Interface)结合 @AutoService 与 ServiceLoader 详解
  4. 一文带你吃透Android中显示Intent与隐式Intent的区别
  5. 一文带你吃透Android View绘制流程与原理详解
注:本文转载自juejin.cn的QING618的文章"https://juejin.cn/post/7483764347040235574"。版权归原作者所有,此博客不拥有其著作权,亦不承担相应法律责任。如有侵权,请联系我们删除。
复制链接
复制链接
相关推荐
发表评论
登录后才能发表评论和回复 注册

/ 登录

评论记录:

未查询到任何数据!
回复评论:

分类栏目

后端 (14832) 前端 (14280) 移动开发 (3760) 编程语言 (3851) Java (3904) Python (3298) 人工智能 (10119) AIGC (2810) 大数据 (3499) 数据库 (3945) 数据结构与算法 (3757) 音视频 (2669) 云原生 (3145) 云平台 (2965) 前沿技术 (2993) 开源 (2160) 小程序 (2860) 运维 (2533) 服务器 (2698) 操作系统 (2325) 硬件开发 (2492) 嵌入式 (2955) 微软技术 (2769) 软件工程 (2056) 测试 (2865) 网络空间安全 (2948) 网络与通信 (2797) 用户体验设计 (2592) 学习和成长 (2593) 搜索 (2744) 开发工具 (7108) 游戏 (2829) HarmonyOS (2935) 区块链 (2782) 数学 (3112) 3C硬件 (2759) 资讯 (2909) Android (4709) iOS (1850) 代码人生 (3043) 阅读 (2841)

热门文章

101
推荐
关于我们 隐私政策 免责声明 联系我们
Copyright © 2020-2024 蚁人论坛 (iYenn.com) All Rights Reserved.
Scroll to Top