Kotlin中的对象,创建安全的单例就一行代码

本文链接
本文结合自己的感受,做一下简单的翻译。原文作者也是《Kotlin for Android developer》的作者。此译文供大家学习参考之用。

Kotlin中的对象,创建安全的单例就一行代码

先说明一个概念:
Kotlin中的object关键字和Java中的Object类, 不是一个东西。所以本文中的对象(object)指的是kotlin中的object关键字。

Kotlin对象(object)作为语言的另外一种关键字(要素),我们Android开发者不熟悉,因为不像Java。实际上,一个对象(object)只是单例实现的类型。因此我们要找和Java相似处,那就是单例模式。我们会在下面对比一下。

单例 VS 对象

听起来Java中的单例不是那么容易实现。

public class Singleton {
 
    private Singleton() {
    }
 
    private static Singleton instance;
 
    public static Singleton getInstance() {
        if (instance == null) {
            instance = new Singleton();
        }
 
        return instance;
    }
}

但是这个代码是危险的,特别是如果他们在不同的线程执行的时候。如果2个线程在同一时刻访问这个单例的话,这个对象会生成2个实例。更安全的代码是如下:

public class Singleton {
 
    private static Singleton instance = null;
 
    private Singleton(){
    }
 
    private synchronized static void createInstance() {
        if (instance == null) {
            instance = new Singleton();
        }
    }
 
    public static Singleton getInstance() {
        if (instance == null) createInstance();
        return instance;
    }
}

如你所见,你需要一些代码创建有效的单例。
那在Kotlin中相等价的是如何实现的呢?

object Singleton

你不需要更多。在菜单『工具』的Kotlin中,你可以使用『显示字节码』工具,及反编译选项。这样,你能看到Kotlin团队是用什么来实现单例模式的(使用静态块来初始化单例)。当你不能肯定在这里发生了什么,我相信你记得使用工具来验证。

对象(object)定义

定义一个对象(Object)和定义一个类一样简单。让我们定义一个实现数据库工具来作为例子:

object MySQLOpenHandler : SQLiteOpenHelper(App.instance, "MyDB", null, 1) {
    
    override fun onCreate(db: SQLiteDatabase?) {
    }
 
    override fun onUpgrade(db: SQLiteDatabase?, oldVersion: Int, newVersion: Int) {
    }
 
}

如你所见,你只需要使用关键词『Object』来替代『Class』,其余一样的。只是对象不能有构造方法,我们通过访问他们来调用相应构造方法。对象的实例会在我们第一次使用的时候创建。因此这是惰性实现的:如果一个对象从来没有被使用,那他的实例也从不会被创建。

Companion 伴生对象(object)

每个类都能实现一个伴生对象(object),是这个类所有实例共有的一个对象。这个有点类似Java中的静态属性。一个实现的例子:

class App : Application() {
    companion object {
        lateinit var instance: App
            private set
    }
 
    override fun onCreate() {
        super.onCreate()
        instance = this
    }
 
}

这个例子我创建了一个类继承了『Application』 并且保存在他唯一实例的『companion 』对象中。『lateinit』修饰表明这个属性不会一开始就有值,但是它会在使用前被赋值(重写会被抛异常)。这个『private set』这个属性不会被外部类赋值。

注意:你可能觉得可以用对象(object)来替代类来实现App,这样Android框架需要实例化这个类。如果你试了,当它启动的时候,你会看到Application抛异常处理,你需要App作为一个类。如果你要访问的话,可以创建一个小点的单例实现。

对象(object)匿名表达式

对象可以作为匿名类实现。
一个例子:

recycler.adapter = object : RecyclerView.Adapter() {
    override fun onBindViewHolder(holder: RecyclerView.ViewHolder?, position: Int) {
    }
 
    override fun getItemCount(): Int {
    }
 
    override fun onCreateViewHolder(parent: ViewGroup?, viewType: Int): RecyclerView.ViewHolder {
    }
 
}

每当你要创建一个接口的内联实现,例如:继承其他类,你会使用上面的表示方式。
但是一个对象也可以作为一个类的代理。你能创建想下面一样创建

val newObj = object {
    var x = "a"
    var y = "b"
}
 
Log.d(tag, "x:${newObj.x}, y:${newObj.y}")

结论

对象从Java6开始,对象对我们来说是一个新的概念,但是很多概念和这个有相关性,因此你们需要快速理解。

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 前言 人生苦多,快来 Kotlin ,快速学习Kotlin! 什么是Kotlin? Kotlin 是种静态类型编程...
    任半生嚣狂阅读 26,289评论 9 118
  • Google在今年的IO大会上宣布,将Android开发的官方语言更换为Kotlin,作为跟着Google玩儿An...
    蓝灰_q阅读 77,007评论 31 489
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 135,009评论 19 139
  • 单例模式(SingletonPattern)一般被认为是最简单、最易理解的设计模式,也因为它的简洁易懂,是项目中最...
    成热了阅读 4,298评论 4 34
  • 窗外雷声大作,狂风暴雨铺天盖地般袭来,站在窗前看着一道道闪电霎时间照亮豆大的雨珠在地面溅起的水泡。母亲顿时占满了我...
    微笑_4b73阅读 134评论 0 0