单例模式
- 单例模式可以分为饿汉式和懒汉式
- 饿汉式会在类装载的时候变完成实例化,如果从未使用过这个实例则会造成内存浪费
- 而懒汉式则是在需要的时候由使用者自行创建实例,这里的问题是如何在多线程环境下保证单例
单例模式的实现方式
饿汉式(静态常量)
注:静态常量位于虚拟机内存的方法区,是线程共享的。
public class Singleton {
private final static Singleton INSTANCE = new Singleton();
private Singleton(){}
public static Singleton getInstance(){
return INSTANCE;
}
}
懒汉式(线程不安全)
实现懒汉式单例,我们首先想到的是构造函数私有化以及获取唯一实例的方法
public class Singleton {
private static Singleton singleton;
private Singleton() {}
public static Singleton getInstance() {
if (singleton == null) {
singleton = new Singleton();
}
return singleton;
}
}
懒汉式(方法锁,线程安全)
于是为了保证线程安全,我们在获取实例的方法上加锁,这样的确可以保证线程安全,但是这里的同步块并非最小粒度
public class Singleton {
private static Singleton singleton;
private Singleton() {}
public static synchronized Singleton getInstance() {
if (singleton == null) {
singleton = new Singleton();
}
return singleton;
}
}
懒汉式(线程不安全)
于是我们理所当然的想到了这种错误的方式,他的问题在于若两个线程同时到达 if (singleton == null) 则会创建两个实例。
public class Singleton {
private static Singleton singleton;
private Singleton() {}
public static Singleton getInstance() {
if (singleton == null) {
synchronized (Singleton.class) {
singleton = new Singleton();
}
}
return singleton;
}
}
懒汉式(双重检查,仍然线程不安全)
接下来的方案仍然是线程不安全的,但是理解他的问题我们需要引入指令重排序的概念。
- 指令重排序 简单的来说就是,由于cpu的效率远高于内存,为了提高代码的执行速度,jvm会在不影响最终结果的前提下,对需要执行语句进行顺序调整。
- 然后我们看代码,在执行instance = new Singleton2() 的时候,jvm大概租了三件事:1分配内存、2创建实例、3将instance指向分配的内存空间。
- 我们无法保证执行顺序一定是123,如果被重排序为了132,在3执行完毕时,另一个线程进入同步块 判断install是否为空时,因instace已经指向了分配的内存空间,会得到false,返回一个未完成实例化的instance。
public class Singleton2 {
private Singleton2(){}
private static Singleton2 instance;
public static synchronized Singleton2 getInstance(){
if(instance == null) {
synchronized (Singleton2.class){
if (instance == null){
instance = new Singleton2();
}
}
}
return instance;
}
}
懒汉式(双重检查禁止重排序)
对于多线程下的重排序问题java给出的解决方案是volatile关键字,他保证了在写操作完成前,不会对其进行读操作。
public class Singleton2 {
private Singleton2(){}
private static volatile Singleton2 instance;
public static synchronized Singleton2 getInstance(){
if(instance == null) {
synchronized (Singleton2.class){
if (instance == null){
instance = new Singleton2();
}
}
}
return instance;
}
}
静态内部类
public class Singleton2 {
private Singleton2(){}
public static class SingletonHolder{
private static final Singleton2 INSTANCE = new Singleton2();
}
public static Singleton2 getInstance(){
return SingletonHolder.INSTANCE;
}
}