}
static {
HungrySin var0 = new HungrySin();
INSTANCE = var0;
}
}
2.懒汉式 doubleCheck
说到延迟加载就直接跳到线程安全且性能较好的doubleCheck吧,若是用java实现会用2层check,第一层判断减轻锁的负担直接判断是否创建过,第二层判断加锁保证线程安全,最后用volatile禁止重排序防止编译器优化导致的线程安全问题。
在kotlin里面也无需这么复杂,直接使用by lazy代理即可实现
class DoubleCheckSin private constructor() {
companion object {
val doubleCheckSin: DoubleCheckSin by lazy {
DoubleCheckSin()
}
}
fun calculate() {
}
}
复制代码
这个要归功于委托和lazy.kt
@kotlin.internal.InlineOnly
public inline operator fun Lazy.getValue(thisRef: Any?, property: KProperty<*>): T = value
复制代码
可以看到上文,把val doubleCheckSin委托给了lazy, get时返回的是lazy.kt的value
重点来看看lazy
layz支持传LazyThreadSafetyMode,有三种可选,默认的是SYNCHRONIZED,也就是线程安全,下面分析一下线程安全的具体实现。
private class SynchronizedLazyImpl(initializer: () -> T, lock: Any? = null) : Lazy, Serializable {
private var initializer: (() -> T)? = initializer
//0._value用volatile禁止指令重排序
@Volatile private var _value: Any? = UNINITIALIZED_VALUE
// final field is required to enable safe publication of constructed instance
private val lock = lock ?: this
//1.这个是上面分析的单例委托的value
override val value: T
get() {
val _v1 = _value
//2.若已经初始化,则直接返回 同java中的第一层判断
if (_v1 !== UNINITIALIZED_VALUE) {
@Suppress(“UNCHECKED_CAST”)
return _v1 as T
}
return synchronized(lock) {
val _v2 = _value
//3.第二层判断,加synchronized关键帧 保证线程安全
if (_v2 !== UNINITIALIZED_VALUE) {
@Suppress(“UNCHECKED_CAST”) (_v2 as T)
} else {
//4.第一次创建,把执行lazy下传进来的lambda 单例这里也就是值的构造构造函数
val typedValue = initializer!!()
_value = typedValue
initializer = null
typedValue
}
}
}
override fun isInitialized(): Boolean = _value !== UNINITIALIZED_VALUE
override fun toString(): String = if (isInitialized()) value.toString() else “Lazy value not initialized yet.”
private fun writeReplace(): Any = InitializedLazyImpl(value)
}
复制代码
上面的代码对应的地方加了注释
-
_value用volatile禁止指令重排序
-
这个是上面分析的单例委托的value
-
若已经初始化,则直接返回 同java中的第一层判断
-
第二层判断,加synchronized关键帧 保证线程安全
-
第一次创建,把执行lazy下传进来的lambda 单例这里也就是值的构造构造函数
由此可以看出lazy的SYNCHRONIZED模式实现和java是完全一样的(除了委托和lambda函数),但是使用koltin by lazy实现doubelCheck单例就就可以少写很多代码
对于by layz其实用的更多的地方是view的延迟加载,道理和单例一样的。第一次访问的时候才执行lambda来初始化,后面就直接取之前创建的对象。
3.静态内部类
这个其实没有取巧的地方了,还是得像java一样要声明一个静态内部类,通过虚拟机的静态内部类加载机制实现线程安全、延迟加载。
【附】相关架构及资料
源码、笔记、视频。高级UI、性能优化、架构师课程、NDK、混合式开发(ReactNative+Weex)微信小程序、Flutter全方面的Android进阶实践技术,和技术大牛一起讨论交流解决问题。
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门,即可获取!
合式开发(ReactNative+Weex)微信小程序、Flutter全方面的Android进阶实践技术,和技术大牛一起讨论交流解决问题。**
[外链图片转存中…(img-ZU9cWgZl-1715784119786)]
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门,即可获取!