【面试题】Java单例设计模式-饿汉式枚举(enum)单例
一、enum关键字
enum关键字是在Java1.5也就是Java SE5之后引入的一个新特性:它通过关键字enum来定义一个枚举类,这个被定义的枚举类继承Enum类,这个枚举类算是一种特殊类,它同样能像其他普通类一样拥有构造器、方法,也能够实现接口,但是它不能再继承其他别的类,因为它的直接父类是Enum类,并且因为它默认的修饰符有final的存在,因此它无法直接派生出其他子类,除非将其使用abstract修饰。
按照《Java编程思想》中的原话来说:关键字enum可以将一组具名的值的有限集合创建为一种新的类型,而这些具名的值可以作为常规的程序组件来使用。
在枚举类出现之前Java是将常量放在接口或是放在普通类当中,然后使用public、static、final去修饰定义的常量,如下两个例子:
public interface Constants2 {
public static final int CONSTANT_1 = 1;
public static final int CONSTANT_2 = 2;
public static final int CONSTANT_3 = 3;
}
public class Constants {
public static final int CONSTANT_1 = 1;
public static final int CONSTANT_2 = 2;
public static final int CONSTANT_3 = 3;
}
在枚举类型出现之后,就可以使用枚举类型来定义常量,这些枚举类型成员_1、_2、_3都默认被public、static、final修饰,语法如下:
public enum Constants {
CONSTANT_1,
CONSTANT_2,
CONSTANT_3
}
但是Java枚举类型输出其常量的时候不像C /C++的枚举那样是数字,输出的是其常量名,如果需要输出其类型成员声明时数字次序的话,需要调用ordinal()方法:
public enum Singleton2 {
SHERLOCK,
WASTON;
}
class Main{
public static void main(String[] args) {
System.out.println(Singleton2.SHERLOCK);
System.out.println(Singleton2.WASTON);
System.out.println(Singleton2.SHERLOCK.ordinal());
System.out.println(Singleton2.WASTON.ordinal());
}
}
输出结果:
SHERLOCK
WASTON
0
1
二、枚举单例的实现
单例模式的特点有以下三个:
- 1、单例类只能有一个实例。
- 2、单例类必须自己创建自己的唯一实例。
- 3、单例类必须给所有其他对象提供这一实例。
我们可以发现枚举类型十分契合以上三个特点,并且我们通过创建枚举类型,可以发现它其中每一个类型成员其实都是Singleton2这个枚举类的一个实例。
public enum Singleton2 {
SHERLOCK
}
class Main{
public static void main(String[] args) {
Singleton2 sherlock = Singleton2.SHERLOCK;
Singleton2 sherlock1 = Singleton2.SHERLOCK;
System.out.println(sherlock == Singleton2.SHERLOCK);
System.out.println(sherlock == sherlock1);
System.out.println(Singleton2.SHERLOCK.getDeclaringClass());
}
}
输出结果:
true
true
class com.sherlock.singleton.Singleton2

利用这个特性,我们就可以通过如下代码创建单例,同时又因为这个特性,决定了它只能属于饿汉式单例模式
public enum Singleton2 {
SHERLOCK;
public void print() {
System.out.println("I am Sherlock!");
}
}
class Main{
public static void main(String[] args) {
Singleton2 sherlock = Singleton2.SHERLOCK;
System.out.println(Singleton2.SHERLOCK.getDeclaringClass());
sherlock.print();
}
}
输出结果如下:
class com.sherlock.singleton.Singleton2 I am Sherlock!
三、枚举单例的优缺点
优点:
(1)能够避免多线程同步问题;
(2)能够防止反序列化重新创建对象;
(3)实现比起其它懒汉式、饿汉式单例来说十分简洁,阅读性好;
缺点:
(1)因为是饿汉式加载,所以会导致枚举实例会长期存在于内存当中;
赞 (0)
