研究一下Scala语言的单例对象(Singleton Objects),为下一篇文章做准备。
static不是keyword
上一篇文章提到过,interface并非Scala语言keyword,能够自由使用。
相同,static在Scala里也没有特殊的含义,也是能够自由使用的,如以下代码所看到的:
单例对象
Java并非完美的面向对象语言,包含非常多缺陷,比方同意static字段和方法,primitive类型。等等。Scala语言在这些方面都有所改进。所以号称是比Java更OO的语言。
既然去掉了statickeyword,那么怎样像Java语言那样。表达类字段或类方法呢?Scala给出的解决方式是:单例对象。Java有一个Math类(java.lang.Math),里头全是static字段和方法,部分代码例如以下所看到的:
public final class Math { private Math() {} // Don't let anyone instantiate this class. public static final double PI = 3.14159265358979323846; public static int abs(int a) { return (a < 0) ?以下我们用Scala语言重写上面的Math类:-a : a; } }
单例对象实现方式
以下看看Scala是怎样实现单例对象的。观察编译结果能够看到,MyMath被编译出两个class:MyMath.class和MyMath$.class。我自己分析了一下这两个class,以下是MyMath.class的反编译结果:
public final class MyMath { public static double PI() { return MyMath$.MODULE$.PI(); } public static int abs(int a) { return MyMath$.MODULE$.abs(a); } }能够得出例如以下结论:
- val字段实际上也被编译成了方法
- 两个方法都是static,并且仅仅是调用MyMath$.MODULE$的对应方法
再来看MyMath$.class的反编译结果:
public final class MyMath$ { public static final MyMath$ MODULE$; private final double PI; static { new MyMath$(); } private MyMath$() { MyMath$.MODULE$ = this; this.PI = 3.14; } public double PI() { return this.PI; } public int abs(int a) { return return (a < 0) ? -a : a; } }就是普通的单例模式,这肯定也就是单例对象这一名称的由来。
使用单例对象
以下这段代码演示了怎样使用单例对象:
看起来和使用Java静态字段或方法没啥差别,以下是反编译之后的main方法代码:
Predef$.MODULE$.println("PI is " + MyMath$.MODULE$.PI()) final int x = -18 final int y = MyMath$.MODULE$.abs(x)
伴随类和伴随对象
上面的样例中,我们定义了名为MyMath的单例对象,实际上。这并最好还是碍我们定义同名的类。
例如以下所看到的:
这样的情况下,单例对象叫做同名类的Companion Object。类叫做单例对象的Companion Class。
假设仅定了单例对象。但未定义同名的类。那么这样的情况下单例对象被叫做Standalone Object。
注意:Companion Class和Object必须定义在同一个.scala文件中。