kotlin 入门到精通 单例模式(六)

}

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)

}

复制代码

上面的代码对应的地方加了注释

  1. _value用volatile禁止指令重排序

  2. 这个是上面分析的单例委托的value

  3. 若已经初始化,则直接返回 同java中的第一层判断

  4. 第二层判断,加synchronized关键帧 保证线程安全

  5. 第一次创建,把执行lazy下传进来的lambda 单例这里也就是值的构造构造函数

由此可以看出lazy的SYNCHRONIZED模式实现和java是完全一样的(除了委托和lambda函数),但是使用koltin by lazy实现doubelCheck单例就就可以少写很多代码

对于by layz其实用的更多的地方是view的延迟加载,道理和单例一样的。第一次访问的时候才执行lambda来初始化,后面就直接取之前创建的对象。

3.静态内部类

这个其实没有取巧的地方了,还是得像java一样要声明一个静态内部类,通过虚拟机的静态内部类加载机制实现线程安全、延迟加载。

【附】相关架构及资料

源码、笔记、视频。高级UI、性能优化、架构师课程、NDK、混合式开发(ReactNative+Weex)微信小程序、Flutter全方面的Android进阶实践技术,和技术大牛一起讨论交流解决问题。

image

《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!
合式开发(ReactNative+Weex)微信小程序、Flutter全方面的Android进阶实践技术,和技术大牛一起讨论交流解决问题。**

[外链图片转存中…(img-VZO56h4m-1715716718728)]

《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!

  • 18
    点赞
  • 26
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值