Kotlin中的5种单例模式示例详解

前言

最近在学习Kotlin这门语言,在项目开发中,运用到了单例模式。因为其表达方式与Java是不同的。所以对不同单例模式的实现进行了分别探讨。主要单例模式实现如下:

  • 饿汉式
  • 懒汉式
  • 线程安全的懒汉式
  • 双重校验锁式
  • 静态内部类式

PS:该篇文章不讨论单例模式的运用场景与各种模式下的单例模式的优缺点。只讨论在Java下不同单例模式下的对应Kotlin实现。

一、饿汉式实现

//Java实现
public class SingletonDemo {
 private static SingletonDemo instance=new SingletonDemo();
 private SingletonDemo(){

 }
 public static SingletonDemo getInstance(){
 return instance;
 }
}
//Kotlin实现
object SingletonDemo

这里很多小伙伴,就吃了一惊。我靠一个object 关键字就完成相同的功能?一行代码?

Kotlin的对象声明

学习了Kotlin的小伙伴肯定知道,在Kotlin中类没有静态方法。如果你需要写一个可以无需用一个类的实例来调用,但需要访问类内部的函数(例如,工厂方法,单例等),你可以把该类声明为一个对象。该对象与其他语言的静态成员是类似的。如果你想了解Kotlin对象声明的更多内容。请点击- - - 传送门

到这里,如果还是有很多小伙伴不是很相信一行代码就能解决这个功能,我们可以通过一下方式查看Kotlin的字节码。

查看Kotlin对应字节码

我们进入我们的Android Studio(我的Android Studio 3.0,如果你的编译器版本过低,请自动升级) 选择Tools工具栏,选择"Kotlin",选择“Show Kotlin Bytecode"

选择过后就会进入到下方界面:

点击"Decompile" 根据字节码得到以下代码

public final class SingletonDemo {
 public static final SingletonDemo INSTANCE;
 private SingletonDemo(){}
 static {
 SingletonDemo var0 = new SingletonDemo();
 INSTANCE = var0;
 }
}

通过以上代码,我们了解事实就是这个样子的,使用Kotlin"object"进行对象声明与我们的饿汉式单例的代码是相同的。

二、懒汉式

//Java实现
public class SingletonDemo {
 private static SingletonDemo instance;
 private SingletonDemo(){}
 public static SingletonDemo getInstance(){
 if(instance==null){
  instance=new SingletonDemo();
 }
 return instance;
 }
}
//Kotlin实现
class SingletonDemo private constructor() {
 companion object {
 private var instance: SingletonDemo? = null
  get() {
  if (field == null) {
   field = SingletonDemo()
  }
  return field
  }
 fun get(): SingletonDemo{
 //细心的小伙伴肯定发现了,这里不用getInstance作为为方法名,是因为在伴生对象声明时,内部已有getInstance方法,所以只能取其他名字
  return instance!!
 }
 }
}

上述代码中,我们可以发现在Kotlin实现中,我们让其主构造函数私有化并自定义了其属性访问器,其余内容大同小异。

如果有小伙伴不清楚Kotlin构造函数的使用方式。请点击 - - - 构造函数
不清楚Kotlin的属性与访问器,请点击 - - -属性和字段

三、线程安全的懒汉式

//Java实现
public class SingletonDemo {
 private static SingletonDemo instance;
 private SingletonDemo(){}
 public static synchronized SingletonDemo getInstance(){//使用同步锁
  if(instance==null){
   instance=new SingletonDemo();
  }
  return instance;
 }
}
//Kotlin实现
class SingletonDemo private constructor() {
 companion object {
  private var instance: SingletonDemo? = null
   get() {
    if (field == null) {
     field = SingletonDemo()
    }
    return field
   }
  @Synchronized
  fun get(): SingletonDemo{
   return instance!!
  }
 }
}

大家都知道在使用懒汉式会出现线程安全的问题,需要使用使用同步锁,在Kotlin中,如果你需要将方法声明为同步,需要添加**@Synchronized**注解。

四、双重校验锁式(Double Check)

//Java实现
public class SingletonDemo {
 private volatile static SingletonDemo instance;
 private SingletonDemo(){}
 public static SingletonDemo getInstance(){
  if(instance==null){
   synchronized (SingletonDemo.class){
    if(instance==null){
     instance=new SingletonDemo();
    }
   }
  }
  return instance;
 }
}
//kotlin实现
class SingletonDemo private constructor() {
 companion object {
  val instance: SingletonDemo by lazy(mode = LazyThreadSafetyMode.SYNCHRONIZED) {
  SingletonDemo() }
 }
}

哇!小伙伴们惊喜不,感不感动啊。我们居然几行代码就实现了多行的Java代码。其中我们运用到了Kotlin的延迟属性 Lazy。

Lazy是接受一个 lambda 并返回一个 Lazy 实例的函数,返回的实例可以作为实现延迟属性的委托: 第一次调用 get() 会执行已传递给 lazy() 的 lambda 表达式并记录结果, 后续调用 get() 只是返回记录的结果。

这里还有有两个额外的知识点。

  • 高阶函数,高阶函数是将函数用作参数或返回值的函数(我很纠结我到底讲不讲,哎)。大家还是看这个 ---高阶函数
  • 委托属性

如果你了解以上知识点,我们直接来看Lazy的内部实现。

Lazy内部实现

public fun <T> lazy(mode: LazyThreadSafetyMode, initializer: () -> T): Lazy<T> =
  when (mode) {
   LazyThreadSafetyMode.SYNCHRONIZED -> SynchronizedLazyImpl(initializer)
   LazyThreadSafetyMode.PUBLICATION -> SafePublicationLazyImpl(initializer)
   LazyThreadSafetyMode.NONE -> UnsafeLazyImpl(initializer)
  }

观察上述代码,因为我们传入的mode = LazyThreadSafetyMode.SYNCHRONIZED,
那么会直接走 SynchronizedLazyImpl,我们继续观察SynchronizedLazyImpl。

Lazy接口

SynchronizedLazyImpl实现了Lazy接口,Lazy具体接口如下:

public interface Lazy<out T> {
  //当前实例化对象,一旦实例化后,该对象不会再改变
 public val value: T
 //返回true表示,已经延迟实例化过了,false 表示,没有被实例化,
 //一旦方法返回true,该方法会一直返回true,且不会再继续实例化
 public fun isInitialized(): Boolean
}

继续查看SynchronizedLazyImpl,具体实现如下:

SynchronizedLazyImpl内部实现

private class SynchronizedLazyImpl<out T>(initializer: () -> T, lock: Any? = null) : Lazy<T>, Serializable {
 private var initializer: (() -> T)? = initializer
 @Volatile private var _value: Any? = UNINITIALIZED_VALUE
 // final field is required to enable safe publication of constructed instance
 private val lock = lock ?: this

 override val value: T
  get() {
   val _v1 = _value
   //判断是否已经初始化过,如果初始化过直接返回,不在调用高级函数内部逻辑
   if (_v1 !== UNINITIALIZED_VALUE) {
    @Suppress("UNCHECKED_CAST")
    return _v1 as T
   }

   return synchronized(lock) {
    val _v2 = _value
    if (_v2 !== UNINITIALIZED_VALUE) {
     @Suppress("UNCHECKED_CAST") (_v2 as T)
    }
    else {
     val typedValue = initializer!!()//调用高级函数获取其返回值
     _value = typedValue //将返回值赋值给_value,用于下次判断时,直接返回高级函数的返回值
     initializer = null
     typedValue
    }
   }
  }
 //省略部分代码
}

通过上述代码,我们发现 SynchronizedLazyImpl 覆盖了Lazy接口的value属性,并且重新了其属性访问器。其具体逻辑与Java的双重检验是类似的。

到里这里其实大家还是肯定有疑问,我这里只是实例化了SynchronizedLazyImpl对象,并没有进行值的获取,它是怎么拿到高阶函数的返回值呢?。这里又涉及到了委托属性。

委托属性语法是: val/var <属性名>: <类型> by <表达式>。在 by 后面的表达式是该 委托, 因为属性对应的 get()(和 set())会被委托给它的 getValue() 和 setValue() 方法。 属性的委托不必实现任何的接口,但是需要提供一个 getValue() 函数(和 setValue()——对于 var 属性)。

而Lazy.kt文件中,声明了Lazy接口的getValue扩展函数。故在最终赋值的时候会调用该方法。

@kotlin.internal.InlineOnly
//返回初始化的值。
public inline operator fun <T> Lazy<T>.getValue(thisRef: Any?, property: KProperty<*>): T = value

五、静态内部类式

//Java实现
public class SingletonDemo {
 private static class SingletonHolder{
  private static SingletonDemo instance=new SingletonDemo();
 }
 private SingletonDemo(){
  System.out.println("Singleton has loaded");
 }
 public static SingletonDemo getInstance(){
  return SingletonHolder.instance;
 }
}
//kotlin实现
class SingletonDemo private constructor() {
 companion object {
  val instance = SingletonHolder.holder
 }

 private object SingletonHolder {
  val holder= SingletonDemo()
 }
}

静态内部类的实现方式,也没有什么好说的。Kotlin与Java实现基本雷同。

最后

附上我写的一个基于Kotlin 仿开眼的项目SimpleEyes(ps: 其实在我之前,已经有很多小朋友开始仿这款应用了,但是我觉得要做就做好。所以我的项目和其他的人应该不同,不仅仅是简单的一个应用。但是,但是。但是。重要的话说三遍。还在开发阶段,不要打我),欢迎大家follow和start

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对我们的支持。

(0)

相关推荐

  • Kotlin 单例实例详解

    Kotlin 单例实例详解 单例的实现方法,可以通过同伴对象,或者 lazy. 示例: class Hello private constructor() { companion object { val instance = Hello() } } 通过 lazy 实现 class Hello private constructor() { private object Holder { val INSTANCE = Hello() } companion object { val insta

  • Kotlin中单例模式和Java的对比浅析

    前言 单例模式,一直以来是我们在日常开发中最常用的一种设计模式,更是面试中非常重要,也非常容易被问到的问题.在日常开发中,大家常用的语言还是Java,但今天我给大家带来的是在Kotlin语言中,单例模式是怎么编写的,并且会对比Java方式,下面话不多说了,来一起看看详细的介绍吧 一.懒人写法(恶汉式) java中 public class Singleton{ public static final Singleton instance = new Singleton(); public Sin

  • Kotlin中的5种单例模式示例详解

    前言 最近在学习Kotlin这门语言,在项目开发中,运用到了单例模式.因为其表达方式与Java是不同的.所以对不同单例模式的实现进行了分别探讨.主要单例模式实现如下: 饿汉式 懒汉式 线程安全的懒汉式 双重校验锁式 静态内部类式 PS:该篇文章不讨论单例模式的运用场景与各种模式下的单例模式的优缺点.只讨论在Java下不同单例模式下的对应Kotlin实现. 一.饿汉式实现 //Java实现 public class SingletonDemo { private static SingletonD

  • Kotlin中的扩展函数与属性示例详解

    前言 Kotlin 中类的扩展方法并不是在原类的内部进行拓展,通过反编译为Java代码,可以发现,其原理是使用装饰模式,对源类实例的操作和包装,其实际相当于我们在 Java中定义的工具类方法,并且该工具类方法是使用调用者为第一个参数的,然后在工具方法中操作该调用者: 理论上来说,扩展函数很简单,它就是一个类的成员函数,不过定义在类的外面.让我们来添加一个方法,来计算一个字符串的最后一个字符: package strings /** * @author :Reginer in 2018/5/22

  • Java设计模式之单例模式示例详解

    目录 0.概述 1.饿汉式 1.1 饿汉式单例实现 1.2 破坏单例的几种情况 1.3 预防单例的破坏 2.枚举饿汉式 2.1 枚举单例实现 2.2 破坏单例 3.懒汉式 4.双检锁懒汉式 5.内部类懒汉式 6.JDK中单例的体现 0.概述 为什么要使用单例模式? 在我们的系统中,有一些对象其实我们只需要一个,比如说:线程池.缓存.对话框.注册表.日志对象.充当打印机.显卡等设备驱动程序的对象.事实上,这一类对象只能有一个实例,如果制造出多个实例就可能会导致一些问题的产生,比如:程序的行为异常.

  • Go 语言单例模式示例详解

    目录 简单单例模式 加锁的单例模式 双check 的单例模式 sync.Once 的单例模式 简单单例模式 单例模式是创建类型的模式,它是为了保证执行期间内只有一个实例.使用 Golang 指针可以很容易的实现单例模式,通过指针保持相同的引用. package singleton type singleton struct{} var instance = &singleton{} func getSingleton() *singleton { return instance } 可以看到整个

  • C++中#include头文件的示例详解

    fstream是C++ STL中对文件操作的合集,包含了常用的所有文件操作.在C++中,所有的文件操作,都是以流(stream)的方式进行的,fstream也就是文件流file stream. 最常用的两种操作为: 1.插入器(<<) 向流输出数据.比如说打开了一个文件流fout,那么调用fout<<"Write to file"<<endl;就表示把字符串"Write to file"写入文件并换行. 2.析取器(>>

  • C#获取本地IP的四种方式示例详解

    1.第一种方式 采用System.Net.Dns的GetHostAddress的方式,具体请看代码: /// <summary> /// 网络不通畅可以获取 /// 不过能获取到具体的IP /// </summary> /// <returns></returns> public static List<IPAddress> GetByGetHostAddresses() { try { IPAddress[] adds = Dns.GetHos

  • node运行js获得输出的三种方式示例详解

    一.通过console.log输出(我最喜欢的) 1.js脚本 1.js var arguments = process.argv.splice(2); //获得入参 var a= arguments[0]; 取第一个 console.log(a) //输出 2.python脚本 test_1.py import os print(os.popen('node 1.js fuck').read()) #打印结果fuck 二.通过文件读写获取 1.js脚本 1.js //npm环境别忘了装了 va

  • 详解Python中生成随机数据的示例详解

    目录 随机性有多随机 加密安全性 PRNG random 模块 数组 numpy.random 相关数据的生成 random模块与NumPy对照表 CSPRNG 尽可能随机 os.urandom() secrets 最佳保存方式 UUID 工程随机性的比较 在日常工作编程中存在着各种随机事件,同样在编程中生成随机数字的时候也是一样,随机有多随机呢?在涉及信息安全的情况下,它是最重要的问题之一.每当在 Python 中生成随机数据.字符串或数字时,最好至少大致了解这些数据是如何生成的. 用于在 P

  • Go语言中循环语句使用的示例详解

    目录 一.概述 1. 循环控制语句 2. 无限循环 二.Go 语言 for 循环 1. 语法 2. for语句执行过程 3. 示例 4. For-each range 循环 三.循环嵌套 1. 语法 2. 示例 四.break 语句 1. 语法 2. 示例 五. continue 语句 1. 语法 2. 示例 六.goto 语句 1. 语法 2. 示例 一.概述 在不少实际问题中有许多具有规律性的重复操作,因此在程序中就需要重复执行某些语句. 循环程序的流程图: Go 语言提供了以下几种类型循环

  • C#面向对象编程中里氏替换原则的示例详解

    目录 里氏替换原则 C# 示例 糟糕的示范 正确的示范 总结 在面向对象编程中,SOLID 是五个设计原则的首字母缩写,旨在使软件设计更易于理解.灵活和可维护.这些原则是由美国软件工程师和讲师罗伯特·C·马丁(Robert Cecil Martin)提出的许多原则的子集,在他2000年的论文<设计原则与设计模式>中首次提出. SOLID 原则包含: S:单一功能原则(single-responsibility principle) O:开闭原则(open-closed principle) L

随机推荐