本文链接
本文结合自己的感受,做一下简单的翻译。原文作者也是《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开始,对象对我们来说是一个新的概念,但是很多概念和这个有相关性,因此你们需要快速理解。