本篇是对现网上流传的 Kotlin 实现静态内部类单例模式的纠正,为了把原理说清楚,文章前奏可能会有些长,熟悉静态内部类单例模式原理的朋友,可以直接跳转到文章最后,直接看结果即可。

最近在整理基础库的时候,需要一个基础类来存储初始化的数据,例如应用的 Application Context,用户的登录 token 等等信息,这些基本都是应用全局类的信息,在应用的整个生命周期都会用到,因此我将这个基础类设计为单例模式来优化性能。

我知道的单例模式就有6钟,饿汉式、懒汉式、线程安全的懒汉式,volatile + 双重校验锁试,静态内部类式,,枚举式。

我挑哪种来用呢?那肯定是性能最好的呀!

一个单例模式性能的好坏,主要考究其在复杂的生产环境下,还能否保证实例的唯一性

可能会碰到的生产环境有以下几种,如果在以下情况下不能保证实例的唯一,那么该单例模式就是有瑕疵的。

  1. 多线程情况下是否会被多次创建。

  2. 反射调用单例类的构造方法时,能否重新创建实例。

  3. 如果单例类实现了 Serializable 接口,其反序列化时生成的实例是否与序列化时的实例是同一个。

枚举式据说是最安全,性能最好的单例模式,Java 不允许反射调用枚举类的构造方法,对枚举类的序列化过程也做了特殊处理,同时枚举类利用语言特性保证了多线程安全。

不过枚举式我用的非常少,个人不太习惯,我挑了使用最顺手的静态内部类式,这种单例模式我用的最多。

然而静态内部类单例模式是有瑕疵的,静态内部类利用了类的加载机制保证了多线程安全,但其构造方法仍然可以通过反射的方式被外部调用。如果类实现了 Serializable 接口,那么其默认的反序列化过程生成的对象与序列化时的对象也是不同的。

虽然有瑕疵,但都是有解决办法滴。

解决反射调用构造方式时可能会重新生成对象的问题,我们只需要在类的构造方法里添加一个标志位校验就可以解决:

class Common private constructor(){companion object {private var flag = false}// 防止反射破坏单例init {if (!flag) {flag = true} else {throw Throwable("SingleTon is being attacked.")}}
}

而反序列化的问题,可以通过在类中声明 readResolve()方法 ,在反序列化返回对象前替换成我们的单例解决。

fun readResolve(): Any {return Common.getInstance()
}

更多解决单例模式瑕疵问题的知识,这里就不展开啦,有兴趣的同学可以自己去网上冲浪一下。

考虑到我的单例类使用场景,不需要实现序列化接口,因此只需要解决反射调用的问题就行了。

瑕疵解决了,那么现在就来给我的单例类实现一个静态内部类的单例模式吧~

静态内部类单例模式,作为一个 Java 老手,用 Java 写我可谓是信手拈来,直接用文档编辑器莽着敲,都不带看的:

// java 实现静态内部类单例
class Common {private static boolean flag = false;// 解决反射调用问题private Common() {if (!flag) {flag = true} else {throw new Throwable("SingleTon is being attachked.")}}public static final Common getInstance() {return CommonSingleTonHolder.sInstance}private static final class CommonSingleTonHolder {private static Common sInstance = new Common();}
}

抬手之间还解决了反射调用的问题~

为什么说静态内部类的单例模式是线程安全的呢? 这里要简单提一下类的加载机制了

简单的说,类加载过程包括五个过程:加载、校验、准备、解析、初始化。

  1. 加载:虚拟机通过类的全限定类名获取类的二进制字节流,通过这个字节流代表的静态存储结构转换为方法区中的运行时数据存储结构,并在堆中生成一个class对象,来作为访问这个运行时数据存储结构的入口。
  2. 校验:虚拟机校验类的字节流文件是否符合虚拟机的规范,是否会对虚拟机的安全造成影响。主要包括文件格式校验,元数据校验,字节码校验,符号引用校验。
  3. 准备:为类中的静态变量分配堆内存,并将其初始化为默认值
  4. 解析:将Class文件中的符号引用转化为指向内存的直接引用。
  5. 初始化:执行类构造器的clinit方法,clinit方法里面包含类的静态变量的赋值操作和静态语句块。

类的静态变量会在准备阶段分配内存,并被初始化为默认值。在初始化阶段,会执行类的<clinit>()方法,执行静态变量的赋值操作和静态语句块。

虚拟机会保证一个类的<clinit>()方法在多线程环境中被正确地加锁、同步,如果多个线程同时去初始化一个类,那么只会有一个线程去执行这个类的<clinit>()方法,其他线程都需要阻塞等待,直到活动线程执行<clinit>()方法完毕。因此说静态内部类的单例在多线程访问时也是线程安全的。

静态内部类单例模式也是一种懒汉模式,只有在执行 Common.getInstance()时才会去加载 CommonSingleTonHolder 类,为 sInstance 静态属性初始化。

Kotlin 实现静态内部类单例模式

前文介绍了这么多都是有关于 Java 的实现方式,用 Kotlin 怎么实现呢?

本着伸手就有,决不自己动手的原则,网上搜索一下,有能直接拿来用的,直接 copy copy~

现网上传的 Kotlin 静态内部类单例模式
// google 第一页搜索内容,截止到 2022 年 1 月 14 日
class SingletonDemo private constructor() {companion object {val instance = SingletonHolder.holder}private object SingletonHolder {val holder = SingletonDemo()}
}

搜索完点击第一条,返回了上面的搜索结果。

喔,看着挺简单的,看看有没其他实现,陆续点击第二条第三条搜索内容,发现一整页的静态内部类单例都是返回这样的结果。

大家都一样,得,就你了。ctrl + c,ctrl + v,改个类名,nice,搞定。

class Common private constructor(){// 防止反射破坏单例init {if (!flag) {flag = true} else {throw Throwable("SingleTon is being attacked.")}}companion object {private var flag = false// 单例val instance = CommonSingletonHolder.holder}/*** 静态内部类单例*/private object CommonSingletonHolder {val holder = Common()}}

通过 Common.instance就可以访问我们的单例了。完美完美,对比一下 Kotlin 与 Java 的实现方式,非常像!

把上面的 Kotlin 代码反编译成 Java,再看,我却觉得有些不对劲。

为了让 Kotlin 反编译生成的代码与 Java 原生的调用方式尽量相同,反编译前,我给 instance 属性加了 @JvmField 注解。

companion object {@JvmFieldval instance = CommonSingletonHolder.holder
}

然后再反编译成 Java 代码,省去与分析无关的代码后:

public final class Common {@JvmField@NotNull// 单例public static final Common instance;private Common() {}// 单例的赋值static {instance = Common.CommonSingletonHolder.INSTANCE.getHolder();}// $FF: synthetic methodpublic Common(DefaultConstructorMarker $constructor_marker) {this();}private static final class CommonSingletonHolder {@NotNullprivate static final Common holder;@NotNullpublic static final Common.CommonSingletonHolder INSTANCE;@NotNullpublic final Common getHolder() {return holder;}static {Common.CommonSingletonHolder var0 = new Common.CommonSingletonHolder();INSTANCE = var0;holder = new Common((DefaultConstructorMarker)null);}}...
}

好像不太对喔!仔细看单例的赋值那一块代码,它是在 static 静态代码块里进行初始赋值的喔。

前文说了,静态内部类本质上也是一种懒汉单例模式,如果 instance 在 静态代码块被初始化,那么 instance 就会在 Common 类加载的过程中就完成初始化,那本质上,它不就变成一种饿汉单例模式了吗?

写个 test case 验证一下 instance 是在什么时候初始化的:

class Common private constructor(){init {JLog.d("Common", "init Common.")}companion object {private var flag = false@JvmFieldval instance = CommonSingletonHolder.holderfun test() {JLog.d("Common", "Common test called.")}}/*** 静态内部类单例*/private object CommonSingletonHolder {val holder = Common()}}// 执行 test() 方法
Common.test()

调用 Common.test() 跑跑看,如果是懒汉模式,那么只要Common没有执行到 CommonSingletonHolder.holder 就不会触发 CommonSingletonHolder 类的加载,更不会走到 val holder = Common() 完成 instance 的初始化。

2022-01-14 14:29:11.330 3500-3500/com.jamgu.common D/Common: init Common.
2022-01-14 14:29:11.330 3500-3500/com.jamgu.common D/Common: Common test called.

Bingo!日志首先打印了 init Common. Instance 在 Common 类加载的时候就被初始化了!,我们实现的居然是一种静态内部类的饿汉模式

啊这,饿汉模式和静态内部类模式都沾点的单例模式,应该叫什么模式??饿静式?听着好像也还说得过去。。

哈哈,开个玩笑,言归正传回来,我们实现的代码既不是静态内部类模式,也不是饿汉模式,那应该怎么修改让它符合一个正宗的静态内部类模式呢?

Easy,类的静态属性和静态代码块,会在类加载的初始化阶段赋值和执行,但类的静态方法不会呀,把单例作为静态方法的返回值就可以完美地让单例懒加载了。

companion object {@JvmStatic// 修改此处fun getInstance() = CommonSingletonHolder.holderfun test() {JLog.d("Common", "Common test called.")}
}

再执行 Common.test(),看看结果:

2022-01-14 14:46:34.914 3862-3862/com.jamgu.common D/Common: Common test called.

没有执行单例的初始化,搞定~

Kotlin 静态内部类单例模式的正确实现方式

最后,一个用 Kotlin 实现的,安全的静态内部类单例模式就崭新出炉了。

class Common private constructor(){// 防止反射破坏单例init {if (!flag) {flag = true} else {throw Throwable("SingleTon is being attacked.")}}companion object {private var flag = false@JvmStaticfun getInstance() = CommonSingletonHolder.holder}/*** 静态内部类单例*/private object CommonSingletonHolder {val holder = Common()}}

现网上传的静态内部类版本不准确噢,仅需要做一些修改,就可以实现真正的静态内部类单例模式了~

@JvmField
val instance = CommonSingletonHolder.holder
// 改成 ----->>>>>
@JvmStatic
fun getInstance() = CommonSingletonHolder.holder

兄dei,如果觉得我写的还不错,麻烦帮个忙呗

Kotlin 静态内部类单例模式的正确实现方式相关推荐

  1. kotlin静态内部类单例

    Kotlin 静态内部类单例模式的正确实现方式: class CommonSingleton private constructor(){companion object {@JvmStaticfun ...

  2. Java与Kotlin的单例模式(霸气.jpg)

    题外话 上一次被人说文章名字取得不霸气,于是这一次我采用了这么霸气的名字,但实际上我是一个很低调的人.设计模式刚入门的小伙伴可以先看看这篇<设计模式入门>,在文章末尾也将列出"设 ...

  3. java静态内部类单例模式_单例模式-静态内部类实现及原理剖析

    以我的经验为例(如有不对欢迎指正),在生产过程中,经常会遇到下面两种情况: 1.封装的某个类不包含具有具体业务含义的类成员变量,是对业务动作的封装,如MVC中的各层(HTTPRequest对象以Thr ...

  4. C# 静态内部类单例模式-静态变量何时初始化

    对于一个类的静态变量何时初始化,大家都有一个普遍的共识,那就是第一次使用该类时,初始化该类的所有静态变量和静态方法. /// <summary>/// 只有在第一次使用到Test1的时候, ...

  5. android动态设置错误页面,Android缺省页的正确打开方式(优雅的处理loading、error、empty...

    Android缺省页的正确打开方式(优雅的处理loading.error.empty Android缺省页的正确打开方式(优雅的处理loading.error.empty各种状态缺省) MultiSt ...

  6. Java静态内部类单例模式读取Properties配置文件

    在Java开发中,可能需要把一些配置参数写入properties配置文件中,在这里介绍一种通过静态内部类单例模式来读取的properties文件的方式. 1.properties文件配置路径 在res ...

  7. python null byte_如何以“正确”的方式处理带有nullbytes的Python unicode字符串?

    问题 PyWin32似乎很乐意将以null结尾的unicode字符串作为返回值.我想用"正确"的方式处理这些字符串.在 假设我得到一个字符串:u'C:\\Users\\Guest\ ...

  8. opengl 贴图坐标控制_材质贴图正确打开方式

    哈喽,各位观众朋友们好鸭~欢迎来到讲道理画图的地方,我是黄玮宁. 最近呀经常有小伙伴来问我那些不同通道的材质贴图该怎么用,而且频率不是一般的高,所以我觉得有必要来说说这些通道贴图的用法了. 视频版(B ...

  9. Optional java 用法_Java8 Optional 的正确使用方式

    1.当我们还在以如下几种方式使用 Optional 时, 就得开始检视自己了 调用 isPresent() 方法时 调用 get() 方法时 Optional 类型作为类/实例属性时 Optional ...

最新文章

  1. 1.STM32中对LED_GPIO_Config()函数的理解(自定义)之流水灯
  2. [收藏]整理了一些T-SQL技巧
  3. 【CEOI2017】Building Bridges【任意坐标斜率优化】【李超线段树】
  4. 关于如何控制一个页面的Ajax读数据只读一次的简单解决办法!
  5. 数字人民币这一年,互联网企业做了什么?
  6. 20145203盖泽双《网络对抗技术》后门原理与实践
  7. springboot 调用Jxbrowser内嵌浏览器
  8. 修改本机域名localhost为任意你想要的名称
  9. QGIS使用栅格图层
  10. Java~基于fluent-hc快速构建Http请求,结合failsafe实现异常重试
  11. (五)AR Foundation实现图片检测(下)
  12. PS磨皮插件portraiture最新版磨皮工具
  13. 10G 网络变压器 10GBASE-T与1000Base-T区别
  14. 图片补全《Globally and locally consistent image completion》
  15. 单链表创建之--头插法创建带头结点的单链表
  16. IDEA高效使用技巧--->IDEA批量修改变量快捷键和全局搜索键
  17. 第三节 树莓派EC20之PPP拨号上网
  18. Greenplum Python专用库gppylib学习——base.py
  19. webservice接口测试,使用SoapUI工具进行接口测试
  20. 再见,我的念青五笔。

热门文章

  1. Cybex在日本:去中心化交易所5大核心功能解决安全问题
  2. Spring中的拦截器
  3. vue怎么看报错在哪一行
  4. ConcurrentHashMap缓存
  5. 相机调试中的色彩问题
  6. hhvm mysql_Linux_在Ubuntu系统上搭建Nginx+HHVM+MySQL开发环境的教程,貌似最近这个HHVM特别火,其主 - phpStudy...
  7. 计算机专业性价比较高的商务本,终于明白了,为何高性价比锐龙本更值得推荐...
  8. centos stream 8 升级和切换python版本的简单方法
  9. VMware虚拟机网络设置(超详细,必看)
  10. python给excel添加超链接_python导出excel(含插入图片,超链接)