单例模式

单例模式

适用于资源占用较多的类,保证一个类只有一个实例即单例。通用的做法就是构造器私有化,提供一个全局的访问点,返回类的实例。

 

1.饿汉式

public class Singleton {
    private static Singleton sin = new Singleton();
    private Singleton(){}
    public static Singleton getSingleton(){
        return sin;
    }
}

这种写法比较简单,实例的初始化在JVM装载类的时候进行,保证了线程的安全性。但是如果从始至终从未使用过这个实例,则会造成内存的浪费。

PS:关于如何保证new对象时候的线程安全性,虚拟机采用了CAS配上失败重试的方式保证更新更新操作的原子性和TLAB两种方式来解决这个问题。(此处参考Java虚拟机2:Java内存区域及对象

 

2.
懒汉式

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;
     }
 }

但是这种写法也存在线程安全的问题:

1.A、B线程同时进入了第一个if判断

2.A首先进入synchronized块,由于instance为null,所以执行instance = new Singleton();

此时正常的执行顺序是:

  1.分配内存空间

  2.初始化对象

  3.instance引用指向内存空间

由于编译器的优化,重新安排执行顺序(但是执行结果不变),可能执行顺序变成:JVM先分配给Singleton实例内存,并将地址赋值给instance变量(注意此时JVM没有开始初始化这个实例),然后A离开了synchronized块。

3.B进入synchronized块,由于instance非null,因此它马上离开了synchronized块并将结果返回给调用该方法的程序。

4.此时B线程使用Singleton实例时,却没有被初始化,会产生空指针异常。    所以把instance加上
volatile就可以解决线程安全的问题。

而volatile 关键字并不是 Java 语言的专属,C语言也有,原意是禁止CPU缓存,例如声明一个volatile变量 volatile int i = 0,表示编译器对这个变量的读写不使用CPU缓存,必须从内存中读取。

这个特性好像跟指令重排没什么关系,但是在1.5后对其语义进行了增强:Happens-Before规则,简单来说就是前面一个操作对于后面操作是可见的,这样一来就可以避免编译优化从而解决上述问题了。

 

点赞