单例模式
单例模式
-
单例模式是一个比较简单的模式,其定义如下:
确保每个类只有一个实例,并且本类实例化对象提供给整个系统(Ensure a class has only one instance, and provide a global point of access to it)。
- 单例模式的通用类图如下:
如上图中的通用类图仅是单例模式中的饿汉模式,其中单例模式还包括懒汉模式、静态内部类、双重校验锁等方式,如下依次通过代码呈现给大家:
public class Singleton { //在Singleton类加载时就回创建singleton对象 private static Singleton singleton = new Singleton(); private Singleton() {} public static Singleton getInstance() { return Singleton.singleton; } }
饿汉模式是基于classloder机制避免了多线程的同步问题,这也使得singleton对象在类装载时就实例化,而我们更希望是在调用getInstance()方法时实例化singleton对象,这样才能达到lazy loading的效果。而懒汉模式就实现了这一愿望。
public class Singleton{ private static Singleton singleton = null; private Singleton() {} //通过synchronized关键字对getInstance方法加锁 public static synchronized Singleton getInstance() { if(singleton == null) { singleton = new Singleton(); } return Singleton.singleton; } }
懒汉模式虽然能够在多线程种很好的工作,而且看起来也具备很好的lazy loading,但是我们并不需要每一次调用getInstance方法都加锁,效率很低。为了解决懒汉模式中效率低下的问题,我们采用静态内部类的方式。
public class Singleton{ private Singleton() {} //静态内部类 private static class SingletonHandler{ private static final Singleton singleton = new Singleton(); } public static Singleton getInstance() { return SingletonHandler.singleton; } }
静态内部类同样利用了classloder的机制来保证初始化instance时只有一个线程,而且实现了当Singleton类被装载时并不立即初始化instance。因为SingletonHandler类没有被主动使用,只有显示通过调用getInstance方法时,才会显示装载SingletonHandler类,从而实例化instance。想象一下,如果实例化instance很消耗资源,我想让他延迟加载,另外一方面,我不希望在Singleton类加载时就实例化,因为我不能确保Singleton类还可能在其他的地方被主动使用从而被加载,那么这个时候实例化instance显然是不合适的。
public class Singleton{ private static Singleton singleton = null; private Singleton() {} public static Singleton getInstance() { if(singleton == null) { synchronized(Singleton.class) { //对Singleton类加锁 if(singleton == null) { singleton = new Singleton(); } } } return singleton; } }
- 单例模式的优点:
- 由于单例模式在内存中只有一个对象,减少了内存开销,当一个对象需要频繁的创建、销毁时,单例模式能提示实例创建时的效率,
- 单例模式减少了内存开销,当一个对象的产生需要比较多的资源时,可以考虑单例模式,创建一个单例对象常驻内存(在Java EE中采用单例模式注意垃圾回收),
- 单例模式避免对资源的多重占用,
- 单例模式可以在系统设置全局访问点,优化和共享资源访问,例如可以设计单例模式负责所有数据表的映射处理。
- 单例模式的缺点:
- 单例模式一般没有接口,扩展困难,
- 单例模式对测试是不利的,如果单例模式没有完成,是不能进行测试的,没有接口也不能使用mock的方式虚拟一个对象,
- 单例模式与单一职责原则冲突。一个类应该只实现一个逻辑,而不关心它是否是单例模式,单例模式把“单例”和业务逻辑融合在一个类中。
-
单例模式的扩展
单例模式是一个类只能产生一个对象的模式,如果一个类可以产生指定个对象,则需要对单例模式做一些变动
- 变动后的类图如下:
-
最佳实践
在Spring中,每个Bean默认就是单例的,这样做的优点就是Spring容器可以管理这些Bean的生命周期,决定什么时候创建出来,什么时候销毁等等。如果采用非单例模式(prototype类型),则Bean初始化后的管理交由J2EE容器,Spring不再跟踪管理Bean的生命周期。