单例模式的升级之路

单例模式:简单的说,就是一个类只能构建一个对象的设计模式。
来看一个最普通的单例:

public class Singleton{
//私有构造函数
private Singleton(){};
//创建单例对象
private static Singleton instance = null;
//静态工厂方法
public static Singleton getinstance(){
    if(instance == null){
        instance = new Singleton();
    }
    return instance;
    }
}

根据以上的单例模式,分析几个关键:
1.私有的构造方法:一个类只能构建一个对象,就不能让它随便去做new操作。

2.初始静态对象:instance是我们的单例对象,也是Singleton类的静态成员。初始值可以是Null,也可以写成new Singleton()。后面解释区别。

3.获取单例对象的方法:getInstance()是获取单例对象的方法。

4.懒汉模式:如果单例初始值是null,还未构建,则构建单例对象并返回。
5.饿汉模式:如果单例对象一开始就被new Singleton()主动构建,则不需要判空操作。

这样理解两种模式:饿汉主动找食物吃,懒汉躺在地上等着人喂

上面的单例模式存在问题:线程不安全
为什么会不安全呢?假设Singleton类刚刚被初始化,线程A和B同时访问getInstance()方法,同时判断instance是否为空时:
这里写图片描述
这时两个线程同时通过了条件判断,都开始执行new操作,这样一来,显然instance被构建了两次。不符合单例模式的设计初衷。下面对代码做个升级。
单例模式升级版:

public class Singleton{
//私有构造函数
private Singleton(){};
//创建单例对象
private static Singleton instance = null;
//静态工厂方法
public static Singleton getinstance(){
    //双重检测机制
    if (instance == null) {      
        //同步锁
         synchronized (Singleton.class){  
           //双重检测机制
           if (instance == null) {     
             instance = new Singleton();
               }
            }
         }
        return instance;
    }
}

下面分析一下升级版的单例模式关键点:
1.Synchronized 同步锁:为了防止new多次 Singleton,所以在new之前加上Synchronized 同步锁,锁住整个类(这里不能使用对象锁)。

2.双重检测机制:进入Synchronized 临界区以后,还要再做一次判空。因为当两个线程同时访问的时候,线程A构建完对象,线程B也已经通过了最初的判空验证,不做第二次判空的话,线程B还是会再次构建instance对象。(两次判空的机制就是双重检测机制)

划重点!!桥黑板!!!划重点了!!!!
看到此处,按照分析来看应该做到了线程安全,但是事实上并没有,还存在漏洞。这是因为 涉及到了JVM编译器的指令重排
什么是指令重排呢?比如一句:Singleton instance = new Singleton();会被编译成如下的JVM指令:
//1:分配对象的内存空间
memory =allocate();
//2:初始化对象
ctorInstance(memory);
//3:设置instance指向刚分配的内存地址
instance =memory;

但是经过JVM和CPU的优化之后,指令会被重排成如下顺序:
//1:分配对象的内存空间
memory =allocate();
//3:设置instance指向刚分配的内存地址
instance =memory;
//2:初始化对象
ctorInstance(memory);

当线程A执行完上述的3时,instance对象还未完成初始化,但已经不再指向null。此时如果线程B抢占到CPU资源,执行 if(instance == null)的结果会是false,从而返回一个没有初始化完成的instance对象。

为了避免这种情况,需要在instance对象前面增加一个修饰符volatile

单例模式终极版:

public class Singleton{
//私有构造函数
private Singleton(){};
//创建volatile修饰的单例对象
private volatile static Singleton instance = null;
//静态工厂方法
public static Singleton getinstance(){
    //双重检测机制
    if (instance == null) {      
        //同步锁
         synchronized (Singleton.class){  
           //双重检测机制
           if (instance == null) {     
             instance = new Singleton();
               }
            }
         }
        return instance;
    }
}

volatile的作用是什么?
可以那么理解:
volatile阻止了被修饰的变量访问前后的指令重排,保证了指令执行顺序;
除此之外,volatile也可以保证线程访问的变量值是主内存中的最新值。

如此,在线程B看来,instance对象的引用要么指向null,要么指向一个初始化完毕的Instance,而不会出现某个中间态,保证了安全。

补充:
上述内容可以创建一个线程安全的单例模式,但是并不代表没有问题,因为利用Java的反射机制,仍然可以new多个对象。如果要避免此问题,采用枚举实现单例模式即可。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值