枚举单例是java中使用枚举提供一个实例对象来实现单例模式的一种新方法,虽然单例模式在java中早已存在,但枚举单例实际上从java5引入枚举作为它的关键特性之后相对来说还是一个新的概念,这篇文章和我先前发表的关于单例的文章有一定的关联性,一篇是java单例模式的10个面试问题,讲的是java面试中关于单例模式常见的10个问题,另一篇是10个关于java枚举的例子,这里面讲了枚举的很多作用。这篇文章是关于我们为什么要使用枚举来实现单例模式,它和传统的单例模式实现方法比起来有哪些好处呢?
下面是我总结的使用枚举实现单例模式的几个原因。另外,如果你喜欢关于设计模式的文章,你也可以看看我发表的关于生成器模式和装饰器模式的文章。
1)枚举单例模式代码简洁
这是迄今为止最大的优点,如果你曾经在java5之前写过单例模式实现代码,那么你会知道即使是使用双检锁你有时候也会返回不止一个实例对象。虽然这种问题通过改善java内存模型和使用volatile变量可以解决,但是这种方法对于很多初学者来说写起来还是很棘手。相比用synchronization的双检索实现方式来说,枚举单例就简单多了。你不相信?比较一下下面的双检索实现代码和枚举实现代码就知道了。
用枚举实现的单例:
这是我们通常写枚举单例的方式,它可能包含实例变量和实例方法,但是简单来说我什么都没用,需要注意的是如果你使用实例方法,你就需要确保方法的线程安全性,避免它会影响对象的状态。通常情况下枚举里面创建实例是线程安全的,但是其它的方法就需要编程者自己去考虑了。
/**
* Singleton pattern example using Java Enumj
*/
public enum EasySingleton{
INSTANCE;
}
代码就这么简单,你可以使用EasySingleton.INSTANCE调用它,比起你在单例中调用getInstance()方法容易多了。
用双检索实现单例:
下面的代码是用双检索实现单例模式的例子,在这里getInstance()方法检查了两次来判断INSTANCE是否为null,这就是为什么叫双检索的原因,记住双检索在java5之前是有问题的,但是java5在内存模型中有了volatile变量之后就没问题了。
/**
* Singleton pattern example with Double checked Locking
*/
public class DoubleCheckedLockingSingleton{
private volatile DoubleCheckedLockingSingleton INSTANCE;
private DoubleCheckedLockingSingleton(){}
public DoubleCheckedLockingSingleton getInstance(){
if(INSTANCE == null){
synchronized(DoubleCheckedLockingSingleton.class){
//double checking Singleton instance
if(INSTANCE == null){
INSTANCE = new DoubleCheckedLockingSingleton();
}
}
}
return INSTANCE;
}
}
你可以访问DoubleCheckedLockingSingleTon.getInstance()来获得实例对象。
现在看看两者创建一个懒加载线程安全的单例需要的代码数量。使用枚举单例模式你只需要一行代码搞定因为枚举实例的创建是线程安全的。
你可能会说比起使用双检索方法还有更好的方法实现单例模式,但是任何一种方法都有它的利和弊,就像我下面例子中展示的我很喜欢的一种在类加载期间初始化静态域的单例实现方式,但是要记住这不是一种懒加载单例方式。
用静态工厂方法实现单例:
这是java中我比较喜欢的一种实现单例模式的方法,由于单例实例是static和final的,当类第一次被加载到内存它就实例化了,所以这种实例的创建方式是线程安全的。
/**
* Singleton pattern example with static factory method
*/public class Singleton{
//initailzed during class loading
private static final Singleton INSTANCE = new Singleton();
//to prevent creating another instance of Singleton
private Singleton(){}
public static Singleton getSingleton(){
return INSTANCE;
}
}
你可以调用Singleton.getInstance()方法来获得实例对象。
2)枚举单例可以自己处理序列化
传统的单例模式的另外一个问题是一旦你实现了serializable接口,他们就不再是单例的了,因为readObject()方法总是返回一个新的实例对象,就像java中的构造器一样。你可以使用readResolve()方法来避免这种情况,通过像下面的例子中这样用单例来替换新创建的实例:
//readResolve to prevent another instance of Singleton
private Object readResolve(){
return INSTANCE;
}
如果你的单例类包含状态的话就变的更复杂了,你需要把他们置为transient状态,但是用枚举单例的话,序列化就不要考虑了。
3)枚举单例是线程安全的
就像第一点提到的,由于枚举实例的创建默认就是线程安全的,你不需要担心双检锁问题。
总结:通过提供序列化和线程安全并且几行代码搞定,说明枚举单例模式是java5之后创建单例最好的方法。你仍然可以使用其它你感觉很流行的方式来创建单例,但是我还是要找一个能够使我信服的观点让我不去使用枚举作为单例,如果你有,请告诉我!(*^__^*)
相关推荐
在软件设计模式中,单例模式是一种常用的创建...源码中的注释将帮助你更好地理解每个部分的作用,从而在实际开发中灵活运用这一模式。记住,在使用单例模式时,应确保其确实符合系统需求,避免滥用导致设计复杂度增加。
单例模式是软件设计模式中的一种基础且广泛应用的模式,它的主要目的是确保一个类只有一个实例,并提供一个全局访问点。在Java中,实现单例模式有多种...在理解了各种实现方式后,才能更好地利用单例模式解决实际问题。
单例模式是软件设计模式中的一种经典模式,用于确保一个类只有一个实例,并提供一个全局访问点。...通过学习和实践这些单例模式的实现,我们可以更好地理解和应用设计模式,提升代码的质量和可维护性。
同时,随着Java技术的发展,枚举单例、基于依赖注入的单例等新型实现方式也逐渐流行,它们提供了更好的解决方案,避免了传统单例模式的一些问题。在阅读《浅议单例模式之线程安全.doc》文档时,可以深入理解这些细节...
枚举在Java中被引入为一个完整的类型,不同于C++中的枚举,这使得Java枚举更加强大且易于使用。以下是关于Java枚举类型的详细解释和相关知识点: 1. 定义枚举: 在Java中,枚举通过关键字`enum`来定义。例如,创建...
在Java中,实现单例模式有多种方法,包括懒汉式(线程不安全)、饿汉式、双重检查锁定(DCL)以及枚举方式。其中,DCL和枚举方式是推荐的,因为它们既保证了线程安全,又避免了不必要的同步开销。 单例模式的Java...
在实际编程中,枚举可以用于实现模式匹配、单例模式等多种设计模式。例如,可以定义一个状态枚举来表示订单的不同状态: ```java public enum OrderStatus { CREATED, PAID, SHIPPED, DELIVERED } ``` 然后,通过...
在早期的Java版本中,没有枚举类型,开发者通常会使用静态常量来模拟枚举的功能。例如: ```java public class Color { public static final Color RED = new Color(); public static final Color GREEN = new ...
枚举常量在`switch`语句中特别有用,它们提供了更好的可读性和安全性,因为无法匹配任何非枚举值。 ```java ShapeType shape = ShapeType.CIRCLE; switch (shape) { case CIRCLE: System.out.println("Circle ...
枚举在集合框架中也有广泛应用,如`java.util.EnumSet`和`java.util.EnumMap`,它们提供了针对枚举类型优化的高效集合实现。 总的来说,Java枚举提供了一种强类型、安全且易于使用的机制来表示固定集合的值。通过...
**设计模式——单例模式** ...如果需要考虑序列化问题,枚举会是更好的选择。总的来说,单例模式的应用旨在提高代码的可维护性和资源利用率,但也要注意避免滥用,以免造成设计复杂度增加和代码难以测试的问题。
在Java中,实现单例模式有多种方法,每种方法都有其特点和适用场景。以下是对这六种常见单例模式实现方式的详细解释: 1. **饿汉式(静态常量)** 这是最简单的单例实现,它在类加载时就创建了实例,线程安全。 `...
它能够提高代码的可读性和可维护性,并且Java中的枚举提供了一些内置的方法来帮助我们更好地管理和操作这些常量。 #### 特性 - **语法格式**:定义枚举的基本语法如下: ```java public enum Color { RED, GREEN...
- **枚举单例**:Java枚举可以自然地实现线程安全的单例,避免了双重检查锁定等问题,且提供了编译时的保证。 总的来说,单例模式在某些特定场景下仍然有用,如全局配置、缓存管理等,但应谨慎使用,因为它可能引入...
总结,Java单例模式在软件设计中有着广泛应用,理解并合理运用不同的实现方式,可以帮助我们更好地控制对象的生命周期,提高系统性能。在实际开发中,需要根据项目需求选择合适的单例实现方式,并注意其可能带来的...
在Java中,考虑到线程安全和简洁性,推荐使用枚举方式实现单例。然而,在某些特定场景下,如需要延迟初始化或控制类加载时机,其他实现方式可能更合适。理解并掌握这些实现方式有助于我们更好地运用单例模式解决实际...
Java单例模式是一种设计模式,它保证一个类只有一个实例,并提供一个全局访问点。这个模式在许多场景下非常有用,比如...通过编写和运行小程序,你可以亲自体验这些不同实现的差异,从而更好地理解和掌握Java单例模式。
在大多数情况下,枚举实现被认为是更推荐的选择,因为它在保证线程安全的同时,也提供了更好的封装性和防止反射攻击的能力。然而,如果对单例的使用有特殊需求,如需要延迟初始化或插件化,那么静态内部类的实现方式...
这种模式在Java编程中非常常见,特别是在需要频繁实例化然后销毁的对象,或者需要共享资源的情况下,单例模式能够提供性能优化和更好的内存管理。 在Java中实现单例模式主要有以下几种方式: 1. 饿汉式(静态常量...
在本资源"MLDN魔乐JAVA_11程序分析、单例设计、static相关应用.rar"中,我们探讨了三个核心的Java编程概念:程序分析、单例设计模式以及静态(static)关键字的应用。这些都是Java开发者必备的知识点,对于理解和...