枚举单例(Enum Singleton)是实现单例模式的一种新方式,尽管单例模式在java中已经存在很长时间了,但是枚举单例相对来说是一种比较新的概念,枚举这个特性是在Java5才出现的,这篇文章主要讲解关于为什么我们应该使用枚举来实现单例模式,它与传统方式实现的单例模式相比较又有哪些优势?
1. 枚举写法简单
写法简单这是它最大的优点,如果你先前写过单例模式,你应该知道即使有DCL(double checked locking) 也可能会创建不止一个实例,尽管在Java5这个问题修复了(jdk1.5在内存模型上做了大量的改善,提供了volatile关键字来修饰变量),但是仍然对新手来说还是比较棘手。对比通过double checked locking 实现同步,枚举单例那实在是太简单了。如果你不相信那么对比下面代码,分别为传统的用double checked locking实现的单例和枚举单例。
枚举实现:
下面这段代码就是声明枚举实例的通常做法,它可能还包含实例变量和实例方法,但是为了简单起见,我并没有使用这些东西,仅仅需要小心的是如果你正在使用实例方法,那么你需要确保线程安全(如果它影响到其他对象的状态的话)。默认枚举实例的创建是线程安全的,但是在枚举中的其他任何方法由程序员自己负责。
你可以通过EnumSingle.INSTANCE来获取单例,比调用getInstance()方法简单多了。
DCL 单例的实现方式
我们可以调用DCLSingleton.getInstance()来获取单例对象的实例,我们也可以把它写成lazy loaded thread-safe单例。
我们从DCL单例和枚举类Enumsingle单例对象可以看出,从代码量相比较,枚举单例更少,因为枚举创建的单例在JVM层面上是可以保证实例的线程安全的。
可能有时候我们会用更好的方式去写单例来替换DCL单例方法,但是每种方法有他的优点和缺点,我们可以通过初始化类加载静态字段,如下所示,但是这种方式不是懒加载形式的单例。
静态工厂实现法:
通过Singleton.getInstance()获取实例。2. 枚举实例创建是线程安全的
枚举单例有序列化和线程安全的保证,而且写法也简单。
网友评论