百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 热门文章 > 正文

聊聊Kotlin单例,从object单例,到带参数的单例如何优雅的封装!

bigegpt 2024-08-28 12:15 3 浏览

一. 序

单例模式是我们在日常编程中,比较常用的设计模式。一个好的单例,必然需要满足唯一性和线程安全性。而 Java 中,关于单例的文章讲解已经很完善了,单例模式已经成为一种编程范式。

在谷歌强推 Kotlin 的今天,不少人使用 Kotlin 时,还带着 Java 的编程思维,并没有有效的利用 Kotlin 的一些特性。如果还用 Java 的编程思想来写 Kotlin 的单例,会有种四不像的感觉。

在 Kotlin 里,想要实现单例模式,只需要将类增加 object 关键字即可,这就是一个线程安全的单例模式,很方便。

但是这存在一个问题,object class 无法实现构造方法,也就是我们无法在初始化的时候,从外部传递一些参数来让这个单例类初始化。

本文就来聊聊 Kotlin 下的单例模式的实现,以及如何优雅的构造一个带参数的单例模式。

二. Kotlin 的单例

2.1 object class 的单例

虽然无法在构造的时候,从外部传递参数,但是 object 关键字依然是 Kotlin 下,最常用的构造单例方法,我们先来了解它的特性。

object 关键字使用起来非常简单,只需要直接作用在 class 上就好。

object SomeSingleton{
 fun sayHi(){}
}

这就是在 Kotlin 下,最简单的单例模式,如果想要有一些初始化的动作,可以放在 init 块中。

object SomeSingleton{
 init{
 // init
 }
 fun sayHi(){}
}

使用方法也非常简单,需要注意的是,在 Kotlin 中调用和 Java 调用存在一些差异。

// Kotlin Language
SomeSingleton.sayHi()
// Java Language
SomeSingleton.INSTANCE.sayHi()

我们知道,Kotlin 和 Java 是可以无缝互通的,而 Kotlin 最终编译的字节码,其实也是可以转成类 Java 的代码。

那我们继续看看 Kotlin 的 object 关键字后,在 Java 中的表现到底如何。通过这种转码的分析,可以便于我们理解 Kotlin 的特性。

借助 AS 的 Tools → Kotlin → Show Kotlin Bytecode,就可以查看 Kotlin 文件的字节码,再点击 Decompile 按钮,就可以将字节码转成 Java 代码

有对比就清晰了,Kotlin 的 object 关键字,在 Java 表现的特点如下:

  1. 为一个 final 类,标记不可变性。
  2. 内部声明一个 static final 的当前类的对象 INSATNCE。
  3. 在静态代码块中,进行 INSTANCE 对象的初始化。

可以看到,在 Kotlin 的 object 中,是使用类的初始化锁来保证线程安全的。

那什么是类的初始化锁?

简单来说, JVM 在类的初始化阶段(即在 Class 被加载后,且被线程使用之前),会执行类的初始化,在初始化期间,JVM 会去获取一个锁。这个锁可以同步多个线程对同一个类的初始化,避免多线程调用时,引发线程安全的问题。

上图很清晰的表明了类的初始化锁的工作流程。

而 Kotlin 中的 object 关键字,就是利用类的初始化锁来保证线程安全的,在我们不需要为单例的初始化传递外部参数的场景下,可以放心使用。

那可能有人担心另一个问题,类加载的时候就初始化构造单例对象,是不是对资源的利用不友好?

这一点问题不大,虚拟机在运行程序的时候,并不是在启动时就将所有的类,都加载进来并初始化完成,而是一种按需加载的策略,在真正使用它的时候,才会初始化。

例如:new Class、调用静态方法、反射、调用 Class.forName() 方法等。这一点可以通过本文介绍的单例实现,在 init 块中输出 Log,看看 Log 何时输出来验证,相关资料很多,就不多说了。

也就是说,通常只有在你真实使用这个类时,它才会真的被虚拟机初始化。当然,不同虚拟机的实现方式不同,这并不是强制的,但是大多数为了性能都会准守此规则。

2.2 传参数的单例

无参单例可以用 object 关键字,但如果想通过一些外部参数初始化单例呢?Kotlin 的 object 是不能有任何构造函数的,所以也无法传递任何参数。

带参单例在 Android 中也是有一些使用场景的,例如 Android 中的 LocalBroadcastManager,就是一个带参的单例模式。

LocalBroadcastManager.getInstance(context).sendBroadcast(intent)

那换个思路想想,在 Java 中,带参数的单例如何实现?通常都会用双重检查锁(Double Checked Locking) + volatile 关键字来解决。

public class DoubleCheckSingleton {
 private volatile static DoubleCheckSingleton sInstance;
 private DoubleCheckSingleton(Context ctx) {
 // init
 }
 public static DoubleCheckSingleton getInstance(Context ctx) {
 if (sInstance == null) {
 synchronized (DoubleCheckSingleton.class) {
 if (sInstance == null) {
 sInstance = new DoubleCheckSingleton(ctx);
 }
 }
 }
 return sInstance;
 }
}

加上 volatile 是为了可见性和禁止重排序,这样就可以保证把参数传递进去的同时,确保线程安全。

不过在 Kotlin 中是没有 volatile 关键字的,取而代之的是 @Volatile 注解,同时需要配合 Kotlin 的伴生对象进行单例模式的构建。

伴生对象可以简单的使用类名作为限定符来调用其方法,类似 Java 中的静态方法。

final class SomeSingleton(context: Context) {
 private val mContext: Context = context
 companion object {
 @Volatile
 private var instance: SomeSingleton? = null
 fun getInstance(context: Context): SomeSingleton {
 val i = instance
 if (i != null) {
 return i
 }
 return synchronized(this) {
 val i2 = instance
 if (i2 != null) {
 i2
 } else {
 val created = SomeSingleton(context)
 instance = created
 created
 }
 }
 }
 }
}

这段代码是直接借鉴的 Kotlin 的 lazy(),lazy 在默认情况下的实现是 SynchronizedLazyImpl,从类名上就能看出来,它使用 synchroinzed 来保证线程安全。

用这样的方式,就可以实现一个可以传参数去构造的单例模式。

2.3 封装一个带参单例

支持传参的单例,我们实现了。但为了实现这个单例,写了 20+ 行代码。每次写单例都要把这一堆代码复制一遍,还挺麻烦,为了使用方便,还可以将其再封装一下。

open class SingletonHolder<out T, in A>(creator: (A) -> T) {
 private var creator: ((A) -> T)? = creator
 @Volatile
 private var instance: T? = null
 fun getInstance(arg: A): T {
 val i = instance
 if (i != null) {
 return i
 }
 return synchronized(this) {
 val i2 = instance
 if (i2 != null) {
 i2
 } else {
 val created = creator!!(arg)
 instance = created
 creator = null
 created
 }
 }
 }
}

用一个支持继承的 open class 加上泛型就可以简单的将其进行封装,此封装方式支持一个参数的构造方法,有需要可以继续扩展或者封装。

class SomeSingleton private constructor(context: Context) {
 init {
 // Init using context argument
 context.getString(R.string.app_name)
 }
 companion object : SingletonHolder<SomeSingleton, Context>(::SomeSingleton)
}

封装成 SingletonHolder 类之后,再想使用单例,关键代码一行就搞定了。

2.4 使用 lazy

前面在介绍带参单例的时候,也提到了lazy(),它是 Kotlin 的一种标准委托,可以接受一个 lambda 并返回一个实例的函数。

如果我们想要延迟初始化,可以使用 lazy() 这个代理来实现,它会在第一次调用 get() 方法时,执行 lazy() 的 lambda 表达式并记录结果,之后再调用 get() 就只会返回之前记录的结果,非常适合延迟初始化的场景。

class SomeSingleton{
 companion object {
 val instance: SomeSingleton by lazy { SomeSingleton() }
 }
}

lazy() 默认情况下,内部就是依赖同步锁(synchronized)来实现的,所以它也是线程安全的。

但是正如我前面提到的,类本身也是按需加载的,调用它的下一步肯定是也需要使用它,所以只要我们正确的使用单例模式,其实没必要使用 lazy(),这里仅做一个介绍,大家知道一下就好了。

三. 小结时刻

本文介绍了在 Kotlin 下,实现单例模式的一些代码技巧,希望对大家有所帮助。最后再简单总结一下。

  1. 无参单例模式,直接使用 Kotlin 的 object 即可,它是依赖类的初始化锁来保证线程安全。
  2. 带参单例模式,可以使用双重检查锁 + @Volatile 来实现,如果嫌麻烦还可以封装成 SingletonHolder。
  3. lazy() 委托确实可以实现延迟加载,但是在单例模式的场景下,不如直接用 object 方便。

本文对你有帮助吗?留言、转发、点赞是最大的支持,谢谢!


在头条号私信我。我会送你一些我整理的学习资料,包含:Android反编译、算法、设计模式、虚拟机、Linux、Kotlin、Python、爬虫、Web项目源码。

相关推荐

得物可观测平台架构升级:基于GreptimeDB的全新监控体系实践

一、摘要在前端可观测分析场景中,需要实时观测并处理多地、多环境的运行情况,以保障Web应用和移动端的可用性与性能。传统方案往往依赖代理Agent→消息队列→流计算引擎→OLAP存储...

warm-flow新春版:网关直连和流程图重构

本期主要解决了网关直连和流程图重构,可以自此之后可支持各种复杂的网关混合、多网关直连使用。-新增Ruoyi-Vue-Plus优秀开源集成案例更新日志[feat]导入、导出和保存等新增json格式支持...

扣子空间体验报告

在数字化时代,智能工具的应用正不断拓展到我们工作和生活的各个角落。从任务规划到项目执行,再到任务管理,作者深入探讨了这款工具在不同场景下的表现和潜力。通过具体的应用实例,文章展示了扣子空间如何帮助用户...

spider-flow:开源的可视化方式定义爬虫方案

spider-flow简介spider-flow是一个爬虫平台,以可视化推拽方式定义爬取流程,无需代码即可实现一个爬虫服务。spider-flow特性支持css选择器、正则提取支持JSON/XML格式...

solon-flow 你好世界!

solon-flow是一个基础级的流处理引擎(可用于业务规则、决策处理、计算编排、流程审批等......)。提供有“开放式”驱动定制支持,像jdbc有mysql或pgsql等驱动,可...

新一代开源爬虫平台:SpiderFlow

SpiderFlow:新一代爬虫平台,以图形化方式定义爬虫流程,不写代码即可完成爬虫。-精选真开源,释放新价值。概览Spider-Flow是一个开源的、面向所有用户的Web端爬虫构建平台,它使用Ja...

通过 SQL 训练机器学习模型的引擎

关注薪资待遇的同学应该知道,机器学习相关的岗位工资普遍偏高啊。同时随着各种通用机器学习框架的出现,机器学习的门槛也在逐渐降低,训练一个简单的机器学习模型变得不那么难。但是不得不承认对于一些数据相关的工...

鼠须管输入法rime for Mac

鼠须管输入法forMac是一款十分新颖的跨平台输入法软件,全名是中州韵输入法引擎,鼠须管输入法mac版不仅仅是一个输入法,而是一个输入法算法框架。Rime的基础架构十分精良,一套算法支持了拼音、...

Go语言 1.20 版本正式发布:新版详细介绍

Go1.20简介最新的Go版本1.20在Go1.19发布六个月后发布。它的大部分更改都在工具链、运行时和库的实现中。一如既往,该版本保持了Go1的兼容性承诺。我们期望几乎所...

iOS 10平台SpriteKit新特性之Tile Maps(上)

简介苹果公司在WWDC2016大会上向人们展示了一大批新的好东西。其中之一就是SpriteKitTileEditor。这款工具易于上手,而且看起来速度特别快。在本教程中,你将了解关于TileE...

程序员简历例句—范例Java、Python、C++模板

个人简介通用简介:有良好的代码风格,通过添加注释提高代码可读性,注重代码质量,研读过XXX,XXX等多个开源项目源码从而学习增强代码的健壮性与扩展性。具备良好的代码编程习惯及文档编写能力,参与多个高...

Telerik UI for iOS Q3 2015正式发布

近日,TelerikUIforiOS正式发布了Q32015。新版本新增对XCode7、Swift2.0和iOS9的支持,同时还新增了对数轴、不连续的日期时间轴等;改进TKDataPoin...

ios使用ijkplayer+nginx进行视频直播

上两节,我们讲到使用nginx和ngixn的rtmp模块搭建直播的服务器,接着我们讲解了在Android使用ijkplayer来作为我们的视频直播播放器,整个过程中,需要注意的就是ijlplayer编...

IOS技术分享|iOS快速生成开发文档(一)

前言对于开发人员而言,文档的作用不言而喻。文档不仅可以提高软件开发效率,还能便于以后的软件开发、使用和维护。本文主要讲述Objective-C快速生成开发文档工具appledoc。简介apple...

macOS下配置VS Code C++开发环境

本文介绍在苹果macOS操作系统下,配置VisualStudioCode的C/C++开发环境的过程,本环境使用Clang/LLVM编译器和调试器。一、前置条件本文默认前置条件是,您的开发设备已...