单例模式
单例模式确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例。
一、单例模式的特点
- 单例类只能有一个实例
- 单例类必须自己创建自己的唯一实例
- 单例类必须给其他对象提供这一实例
二、单例模式的具体实现
饿汉模式:
public class EagerSingleton { private static EagerSingleton EAGER_SINGLETON = new EagerSingleton(); // 私有化 private EagerSingleton() { } private static EagerSingleton getInstance() { return EAGER_SINGLETON; } }可以看出,在单例类被加载时,静态变量EAGER_SINGLETON就会实例化,加载速度快,但是获取对象的速度慢。而且这种基于类加载的机制避免了多线程同步的问题。
懒汉模式:
public class Singleton { private static Singleton instance; private Singleton() { } public static Singleton getInstance() { if (instance == null) { instance = new Singleton(); } return instance; } }
懒汉模式提供了一个静态变量,第一次调用getInstance方法时,单例类会进行初始化,在多线程下不能正常工作。
懒汉模式(线程安全):
public class Singleton { private static Singleton instance; private Singleton() { } public static synchronized Singleton getInstance() { if (instance == null) { instance = new Singleton(); } return instance; } }
这种模式相对于一般的懒汉模式可以在多线程中正常工作,但是每次调用都需要进行同步,这样可能在多次调用时增加不必要的开销。
双重检查模式:
public class Singleton { private static Singleton instance; private Singleton() { } public static Singleton getInstance() { if (instance == null) { synchronized (Singleton.class) { if (instance == null) { instance = new Singleton(); } } } return instance; } }
进行两次是否为空的判断,第一次为了减少不必要的同步,第二次当Singleton为null时,才创建实例。这种写法可以减少资源的消耗和多余的同步,解决线程安全问题。