Android经典面试题之Kotlin中object关键字实现的是什么类型的单例模式?原理是什么?怎么实现双重检验锁单例模式?

简介: Kotlin 单例模式概览在 Kotlin 中,`object` 关键字轻松实现单例,提供线程安全的“饿汉式”单例。例如:要延迟初始化,可使用 `companion object` 和 `lazy` 委托:对于参数化的线程安全单例,结合 `@Volatile` 和 `synchronized`

本文首发于公众号“AntDream”,欢迎微信搜索“AntDream”或扫描文章底部二维码关注,和我一起每天进步一点点

object关键字实现单例模式

在 Kotlin 中实现单例模式非常简单,因为它提供了 object 关键字,可以用来创建单例对象。这里是一个简洁的示例和详细的解释:

定义单例对象

直接使用 object 关键字创建单例对象,这是最简单的方法。这个方法不需要编写额外的代码来确保该对象只有一个实例。

object Singleton {
    var someProperty: String = "default"

    fun doSomething() {
        println("Doing something with property: $someProperty")
    }
}

使用单例对象

可以像使用普通对象一样使用单例对象:

fun main() {
    // 修改属性
    Singleton.someProperty = "Hello, Kotlin"

    // 调用方法
    Singleton.doSomething()  // 输出: Doing something with property: Hello, Kotlin
}

object关键字原理

Kotlin中的object关键字用于声明一个单例对象。这个对象在第一次访问时会被实例化,之后所有对该对象的引用都指向同一个实例。

字节码

public final class Singleton {
   
   @NotNull
   private static String someProperty;
   @NotNull
   public static final Singleton INSTANCE;

   @NotNull
   public final String getSomeProperty() {
   
      return someProperty;
   }

   public final void setSomeProperty(@NotNull String var1) {
   
      Intrinsics.checkNotNullParameter(var1, "<set-?>");
      someProperty = var1;
   }

   public final void doSomething() {
   
      String var1 = "Doing something with property: " + someProperty;
      System.out.println(var1);
   }

   private Singleton() {
   
   }

   static {
   
      Singleton var0 = new Singleton();
      INSTANCE = var0;
      someProperty = "default";
   }
}

其实现原理可以通过以下几个步骤进行解析:

1、 Class 静态初始化块

  • 在Kotlin的单例对象被第一次引用时,它会触发一个静态初始化块来创建这个对象的实例。这类似于Java中的静态初始化块。

2、 线程安全

  • object关键字生成的单例是线程安全的。这是通过JVM的类加载机制保证的,JVM会确保类的静态初始化块在多线程环境中只会被执行一次。

3、 饿汉式单例

  • 从严格意义上来说,object关键字生成的单例更接近于“饿汉式单例”模式,因为该实例会在类加载时被创建并初始化。

Kotlin中的懒汉式单例

懒汉式单例是一种在第一次需要时才创建实例的单例模式,搭配Kotlin的lazy委托可以简单实现:

class LazySingleton private constructor() {
    companion object {
        val instance: LazySingleton by lazy { LazySingleton() }
    }
}

Kotlin中的双重检验锁单例模式

如果你需要传递参数来初始化单例,可以考虑双重检验锁单例模式

双重检验锁单例模式可以确保在多线程环境中的高效及线程安全,虽然在Kotlin中不太常用,但也可以通过@Volatile 关键字以及synchronized关键字实现:

class SingletonWithParams private constructor(val someProperty: String) {

    init {
        println("SingletonWithParams is initialized with someProperty: $someProperty")
    }

    companion object {
        @Volatile private var instance: SingletonWithParams? = null

        fun getInstance(property: String): SingletonWithParams {
            return instance ?: synchronized(this) {
                instance ?: SingletonWithParams(property).also { instance = it }
            }
        }
    }

    fun doSomething() {
        println("Doing something with property: $someProperty")
    }
}

使用带参数的单例

fun main() {
    val singleton = SingletonWithParams.getInstance("Hello, Kotlin with Params")
    singleton.doSomething()  // 输出: Doing something with property: Hello, Kotlin with Params

    // 尝试获取另一个实例
    val anotherSingleton = SingletonWithParams.getInstance("Another Value")
    anotherSingleton.doSomething()  // 输出: Doing something with property: Hello, Kotlin with Params
}

在上面的例子中,通过 companion object 提供一个静态方法 getInstance 来获取单例实例,并且通过 @Volatilesynchronized 确保线程安全。

总结

Kotlin 提供了多种方便且简洁的方法来实现单例模式:

1、 Object 关键字:最简单的方式,适用于没有参数的单例。
2、 伴生对象以及自定义静态方法:适用于需要初始化参数或自定义初始化逻辑的单例。也就是Java中的DCL单例

根据实际需求选择合适的方法,可以让你的代码更加简洁和有效。


欢迎关注我的公众号AntDream查看更多精彩文章!

目录
相关文章
|
30天前
|
NoSQL Java API
美团面试:Redis锁如何续期?Redis锁超时,任务没完怎么办?
在40岁老架构师尼恩的读者交流群中,近期有小伙伴在面试一线互联网企业时遇到了关于Redis分布式锁过期及自动续期的问题。尼恩对此进行了系统化的梳理,介绍了两种核心解决方案:一是通过增加版本号实现乐观锁,二是利用watch dog自动续期机制。后者通过后台线程定期检查锁的状态并在必要时延长锁的过期时间,确保锁不会因超时而意外释放。尼恩还分享了详细的代码实现和原理分析,帮助读者深入理解并掌握这些技术点,以便在面试中自信应对相关问题。更多技术细节和面试准备资料可在尼恩的技术文章和《尼恩Java面试宝典》中获取。
美团面试:Redis锁如何续期?Redis锁超时,任务没完怎么办?
|
2月前
|
消息中间件 安全 前端开发
面试官:单核服务器可以不加锁吗?
面试官:单核服务器可以不加锁吗?
49 4
面试官:单核服务器可以不加锁吗?
|
2月前
|
存储 缓存 安全
【Java面试题汇总】多线程、JUC、锁篇(2023版)
线程和进程的区别、CAS的ABA问题、AQS、哪些地方使用了CAS、怎么保证线程安全、线程同步方式、synchronized的用法及原理、Lock、volatile、线程的六个状态、ThreadLocal、线程通信方式、创建方式、两种创建线程池的方法、线程池设置合适的线程数、线程安全的集合?ConcurrentHashMap、JUC
【Java面试题汇总】多线程、JUC、锁篇(2023版)
|
30天前
|
存储 Kubernetes 架构师
阿里面试:JVM 锁内存 是怎么变化的? JVM 锁的膨胀过程 ?
尼恩,一位经验丰富的40岁老架构师,通过其读者交流群分享了一系列关于JVM锁的深度解析,包括偏向锁、轻量级锁、自旋锁和重量级锁的概念、内存结构变化及锁膨胀流程。这些内容不仅帮助群内的小伙伴们顺利通过了多家一线互联网企业的面试,还整理成了《尼恩Java面试宝典》等技术资料,助力更多开发者提升技术水平,实现职业逆袭。尼恩强调,掌握这些核心知识点不仅能提高面试成功率,还能在实际工作中更好地应对高并发场景下的性能优化问题。
|
2月前
|
ARouter 测试技术 API
Android经典面试题之组件化原理、优缺点、实现方法?
本文介绍了组件化在Android开发中的应用,详细阐述了其原理、优缺点及实现方式,包括模块化、接口编程、依赖注入、路由机制等内容,并提供了具体代码示例。
45 2
|
1月前
|
Java 调度 Android开发
Android面试题之Kotlin中async 和 await实现并发的原理和面试总结
本文首发于公众号“AntDream”,详细解析了Kotlin协程中`async`与`await`的原理及其非阻塞特性,并提供了相关面试题及答案。协程作为轻量级线程,由Kotlin运行时库管理,`async`用于启动协程并返回`Deferred`对象,`await`则用于等待该对象完成并获取结果。文章还探讨了协程与传统线程的区别,并展示了如何取消协程任务及正确释放资源。
23 0
|
3月前
|
缓存 Java
【多线程面试题二十三】、 说说你对读写锁的了解volatile关键字有什么用?
这篇文章讨论了Java中的`volatile`关键字,解释了它如何保证变量的可见性和禁止指令重排,以及它不能保证复合操作的原子性。
|
3月前
|
Java
【多线程面试题二十二】、 说说你对读写锁的了解
这篇文章讨论了读写锁(ReadWriteLock)的概念和应用场景,强调了读写锁适用于读操作远多于写操作的情况,并介绍了Java中`ReentrantReadWriteLock`实现的读写锁特性,包括公平性选择、可重入和可降级。
|
3月前
|
存储 Java
【IO面试题 四】、介绍一下Java的序列化与反序列化
Java的序列化与反序列化允许对象通过实现Serializable接口转换成字节序列并存储或传输,之后可以通过ObjectInputStream和ObjectOutputStream的方法将这些字节序列恢复成对象。
|
7天前
|
存储 算法 Java
大厂面试高频:什么是自旋锁?Java 实现自旋锁的原理?
本文详解自旋锁的概念、优缺点、使用场景及Java实现。关注【mikechen的互联网架构】,10年+BAT架构经验倾囊相授。
大厂面试高频:什么是自旋锁?Java 实现自旋锁的原理?