阅读:Null检查机制及lateinit与by lazy(委托机制)的应用
https://blog.csdn.net/Shenpibaipao/article/details/76974461
lateinit 和 lazy 是 Kotlin 中的两种不同的延迟初始化的实现
lateinit 只用于变量 var,而 lazy 只用于常量 val
lazy 应用于单例模式(if-null-then-init-else-return),而且当且仅当变量被第一次调用的时候,委托方法才会执行。
lazy()
是接受一个 lambda 并返回一个 Lazy <T>
实例的函数,返回的实例可以作为实现延迟属性的委托: 第一次调用 get()
会执行已传递给 lazy()
的 lambda 表达式并记录结果, 后续调用 get()
只是返回记录的结果。
val lazyValue: String by lazy {
println("computed!")
"Hello"
}
fun main(args: Array<String>) {
println(lazyValue)
println(lazyValue)
}
打印结果
computed!
Hello
Hello
比如这样的常见操作,只获取,不赋值,并且多次使用的对象
private val mUserMannager: UserMannager by lazy {
UserMannager.getInstance()
}
再比如acitivity中控件初始化的操作,一般传统的进入界面就初始化所有的控件,而使用懒加载,只有用到时才会对控件初始化
//kotlin 封装:
fun <V : View> Activity.bindView(id: Int): Lazy<V> = lazy {
viewFinder(id) as V
}
//acitivity中扩展调用
private val Activity.viewFinder: Activity.(Int) -> View?
get() = { findViewById(it) }
//在activity中的使用姿势
val mTextView by bindView<TextView>(R.id.text_view)
mTextView.text="执行到我时,才会进行控件初始化"
或
val mLayoutContext:ViewGroup by lazy{
findViewById(R.id.mLayoutContent) as ViewGroup
}
lateinit(延迟初始化属性)
一般地,属性声明为非空类型必须在构造函数中初始化(我们知道,kotlin中默认是空安全的,任何属性的声明都必须有初始化值,如果支持可空”?”,才能把属性声明为null)。然而这样经常不方便,例如:属性可以通过依赖注入来初始化,或者单元测试的setup方法中初始化,这种情况下,你不能在构造函数内提供一个非空初始器,但你仍然想在类体中引用该属性时避免空检查。为处理这种情况,我们可以使用lateinit
val nameA:String //报红线 提示如下
Property must be initialized or be abstract
lateinit var name:String //lateinit可以避免这种情况
该修饰只能用于类体中(不是在主构造函数中)声明的var属性,注意是var(可变属性)并且仅当该属性没有自定义getter或setter时,该属性必须是非空类型,并且不能是原生类型。
在初始化前访问一个lateinit属性会抛出一个特定异常,该异常明确标识该属性被访问及它没有初始化的事实。