在软件设计中,设计模式是针对某一普遍问题的通用解决方案。解决方案翻译成代码,然后在出现问题的时候,代码可以运用在不同的状况下。设计模式一书将模式划分为以下几个主要区域:创建、结构和行为。创建模式描述了对象是如何创建的,结构模式指导如何连接和合并对象。行为模式则描述了对象之间的通信机制。本文主要讲述单例模式。
在软件设计中,设计模式是针对某一普遍问题的通用解决方案。解决方案转换成代码,然后在出现问题的时候,代码可以运用在不同的状况下。设计模式一书将模式划分为以下几个主要区域:创建、结构和行为。创建模式描述了对象是如何创建的,结构模式指导如何连接和合并对象。行为模式则描述了对象之间的通信机制。本文主要讲述单例模式。
单例模式是创建模式中普遍采用的一种。使用单例模式可以确保某个类只会有一个实例被创建。单例模式是通过下面的思想来保证的:不让类以外的任何事物创建对象的实例。通常来讲,单例可以缩减内存的需求。实现方式也有很多种。
假如你知道将要创建的实例是一个子类,那么将父类声明为抽象类并提供一个方法来获得当前的实例。在AWT包中,Toolkit类就是一个典型的代表。Toolkit的构造器是public的。
public Toolkit()
并且类具有一个方法getDefaultToolkit()用于获得特定的子类。在这里,子类是和平台相关的。
public static Toolkit getDefaultToolkit()
在Linux平台上,特定的子类是sun.awt.X11.XToolkit。然而,你并不需要知道这些细节,因为访问的时候我们是通过子类的抽象父类Toolkit实现的。
Collator类是单例模式中另一个例子,只是实现略有不同。它提供了两个getInstance()方法。无参数的版本得到默认locale的Collator。也可以通过传递一个参数来获得指定locale的Collator。通过同样的locale参数多次获得的Collator实例是相同的。Collator的构造器是protected类型的。在J2SE标准类库中,我们还可以找到很多这样的例子。
我们可能会认为限制一个类的构造器的访问会自动将这个类设计为单例模式的,但是并非如此。Calendar就是这样一个例子,Calendar的构造器是protected的,提供了一个getInstance()方法来获得这个类的实例,每次调用getInstance()都会创建一个新的实例。因此它不是单例模式的。
当创建自己的单例类的时候,确保只有一个实例被创建:
public class MySingleton {
private static final MySingleton INSTANCE =
new MySingleton();
private MySingleton() {
}
public static final MySingleton getInstance() {
return INSTANCE;
}
}
静态方法getInstance()返回这个类的一个实例。注重即使这个实例需要是子类,也无须修改API。
一般来说,不需要提供一个getInstance()方法,因为INSTANCE变量可以声明为public的。但是,getInstance()方法可以提供更好的灵活性,尤其是以后系统的设计发生变化的时候。出色的虚拟机实现可以将getInstance()方法进行内联。
编写一个好的单例模式的类并非这么简单,假如需要使得自己的单例类是可序列化的,那么必须提供一个readResolve()方法:
/**
* Ensure Singleton class
*/
private Object readResolve() throws ObjectStreamException {
return INSTANCE;
}
提供readResolve()方法后,反序列化的时候将只有一个对象产生,无论调用了多少次getInstance()方法。假如不提供readResolve()方法,当反序列化的时候,每次都会创建一个新的对象实例。
假如只需要使用一个单独的资源,并且需要共享这个单独资源的状态信息的时候,单例模式是非常有用的。在设计的时候就标记好单例模式的需求可以简化开发。然而,有些时候我们意识不到需要使用单例模式直到系统出现了性能问题,
这个时候我们就需要重构代码/例如,你可能发现系统的性能在下滑,原因是程序中重复的创建了同一个类的很多对象,通过应用单例模式则可以很好的避免创建通常的对象。这可以减少系统用来创建对象的时间,也可以节省垃圾收集器用来释放这些实例的时间。
总的来说,假如不想创建一个类的多个实例的时候就使用单例模式。假如构造器中不需要其他的操作,那么就提供一个空的私有构造器,假如需要子类的话就提供一个protected类型的。否则,默认情况下,系统会提供一个公共的构造器。
需要注重的是在一个给定的类装载器中单例模式保证是唯一的。假如在多个不同的企业容器中使用通常的类的时候,那么需要为每个容器提供一个实例。