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查看更多精彩文章!

目录
相关文章
|
3月前
|
安全 Android开发 Kotlin
Android经典实战之SurfaceView原理和实践
本文介绍了 `SurfaceView` 这一强大的 UI 组件,尤其适合高性能绘制任务,如视频播放和游戏。文章详细讲解了 `SurfaceView` 的原理、与 `Surface` 类的关系及其实现示例,并强调了使用时需注意的线程安全、生命周期管理和性能优化等问题。
176 8
|
1月前
|
缓存 Java 数据库
Android的ANR原理
【10月更文挑战第18天】了解 ANR 的原理对于开发高质量的 Android 应用至关重要。通过合理的设计和优化,可以有效避免 ANR 的发生,提升应用的性能和用户体验。
124 56
|
22天前
|
安全 IDE Java
Kotlin教程笔记(3) - 空类型和智能类型转换
Kotlin教程笔记(3) - 空类型和智能类型转换
22 1
|
1月前
|
安全 IDE Java
Kotlin教程笔记(3) - 空类型和智能类型转换
Kotlin教程笔记(3) - 空类型和智能类型转换
|
1月前
|
安全 IDE Java
Kotlin教程笔记(3)- 空类型和智能类型转换
本教程详细讲解了Kotlin中的空类型、非空与可空类型、可空类型操作符、安全调用操作符、Elvis运算符、非空断言运算符以及智能类型转换等内容,帮助开发者更好地理解和使用Kotlin的空安全机制。适合希望深入了解Kotlin语法的开发者。快速入门请参考“简洁”系列教程。
35 3
|
2月前
|
安全 IDE Java
Kotlin 学习笔记- 空类型和智能类型转换
Kotlin 学习笔记聚焦于空类型和智能类型转换,深入解析非空与可空类型、安全调用操作符、Elvis 运算符、非空断言运算符及智能类型转换等内容,助你高效掌握 Kotlin 语言特性,避免 NullPointException 异常,提升代码质量。
31 2
|
2月前
|
安全 IDE Java
Kotlin 学习笔记- 空类型和智能类型转换
Kotlin 学习笔记- 空类型和智能类型转换
51 2
|
2月前
|
JavaScript
Vue 的响应式原理中 Object.defineProperty 有什么缺陷
Vue 的响应式原理主要依赖于 `Object.defineProperty`,但该方法存在一些缺陷:无法检测到对象属性的添加和删除,且对大量数据进行代理时性能较差。Vue 3 中改用了 Proxy 来解决这些问题。
|
2月前
|
XML 前端开发 Android开发
Android View的绘制流程和原理详细解说
Android View的绘制流程和原理详细解说
48 3
|
2月前
|
Python
通过 type 和 object 之间的关联,进一步分析类型对象
通过 type 和 object 之间的关联,进一步分析类型对象
64 3
下一篇
DataWorks