创建单例对象为什么要加volatile?

  • 先来一段创建单例对象的代码
public class Singleton {
    private volatile static Singleton singleton;
    private Singleton(){
    }
    public static Singleton getInstance(){
    //另一个线程进来刚好singleton不为null
        if(singleton == null){
            synchronized(Singleton.class){
                if(singleton == null){
                // 这里不加volatile有可能导致指令重排序
                    singleton = new Singleton();
                }
            }
        }
        //返回
        return singleton;
    }
}
 
  • 那么为什么要加volatile呢?
  • 首先singleton = new Singleton();这个创建对象的操作内部要执行很多个指令。如下:
    (1) 分配一块内存。
    (2)给这个成员变量赋个默认值,比如0。
    (3)初始化,调用对象构造方法。
    (4)把对象引用传递给singleton 。
  • 那么问题就来了,在不加volatile的时候,这些指令在执行的时候有可能会被优化,也就是可能会被指令重排序;有可能进行完第二步,就直接进行第四步了,这时候singleton就成了个半初始化对象,而另一个线程进来刚好singleton不为null,就直接返回这个半初始化对象了。

posted on 2019-07-14 10:33  愤怒的苹果ext  阅读(27)  评论(0编辑  收藏  举报

导航