单例模式
单例(Singleton)模式的定义:指一个类只有一个实例,且该类能自行创建这个实例的一种模式。例如,Windows 中只能打开一个任务管理器,这样可以避免因打开多个任务管理器窗口而造成内存资源的浪费,或出现各个窗口显示内容的不一致等错误。
单例模式有 3 个特点:
- 单例类只有一个实例对象;
- 该单例对象必须由单例类自行创建;
- 单例类对外提供一个访问该单例的全局访问点;
单例模式的结构
- 单例类:包含一个实例且能自行创建这个实例的类。
- 访问类:使用单例的类。
实现单例模式
public class Singleton
{
private static final Singleton instance=new Singleton();
public static Singleton getInstance()
{
return instance;
}
}
上面代码有什么问题?
public class Singleton
{
private static final Singleton instance=new Singleton();
private LazySingleton(){}
public static Singleton getInstance()
{
return instance;
}
}
上面这种也称为饿汉式单例
;
public class LazySingleton
{
private static volatile LazySingleton instance=null; //保证 instance 在所有线程中同步
private LazySingleton(){} //private 避免类在外部被实例化
public static synchronized LazySingleton getInstance()
{
//getInstance 方法前加同步
if(instance==null)
{
instance=new LazySingleton();
}
return instance;
}
}
上面这种也称为懒汉式单例
。
class Singleton {
private static volatile Singleton1 singleton = null;
private Singleton(){}
private static Singleton getSingleton() {
if (singleton == null) {
synchronized (Singleton.class) {
if (singleton == null) {
singleton = new Singleton();
}
}
}
return singleton;
}
}
上面就是所说的"双检锁
(DCL)"。
pbulic class Singleton {
private Singleton(){}
public static Singleton getSingleton() {
return Holder.singleton;
}
private static class Holder {
private static Singleton singleton = new Singleton();
}
}
上面是利用内部类持有内部对象的方式实现,理论依据是对象初始化过程中隐含的初始化锁。和前面的双检锁实现都能保证线程安全。
单例模式的应用场景
- 在应用场景中,某类只要求生成一个对象的时候,如一个班中的班长、每个人的身份证号等。
- 当对象需要被共享的场合。由于单例模式只允许创建一个对象,共享该对象可以节省内存,并加快对象访问速度。如 Web 中的配置对象、数据库的连接池等。
- 当某类需要频繁实例化,而创建的对象又频繁被销毁的时候,如多线程的线程池、网络连接池等。
单例模式的扩展
单例模式可扩展为有限的多例(Multitcm)模式,这种模式可生成有限个实例并保存在 ArmyList 中,客户需要时可随机获取,其结构图如图