java 双重检查 安全,Java双重检查锁单例真的线程安全吗?

d02176825ffda92d7f7455aec5d71591.png

相信大多数同学在面试当中都遇到过手写单例模式的题目,那么如何写一个完美的单例是面试者需要深究的问题,因为一个严谨的单例模式说不定就直接决定了面试结果,今天我们就要来讲讲看似线程安全的双重检查锁单例模式中可能会出现的指令重排问题。

双重检查锁单例模式

乍一看下面单例模式没啥问题,还加了同步锁保证线程安全,从表面上看确实看不出啥问题,当在同一时间多个线程同时执行该单例时就会出现JVM指令重排的问题,从而可能导致某一个线程获取的single对象未初始化对象。

publicclass Single {

private staticSingle single;

private Single() {

}

publicstaticSingle getInstance() {

if(null== single) {

synchronized (Single.class) {

if(null== single) {

single = new Single();

}

}

}

returnsingle;

}

}

问题前因后果

其实single = new Single()这段代码并不具备原子性,从代码层面上来说确实没问题,但是如果了解JVM指令的就知道其实在执行这句代码的时候在JVM中是需要执行三个指令来完成的,如下:

//1:分配对象的内存空间

memory = allocate();

//2:初始化对象

ctorInstance(memory);

//3:设置instance指向刚分配的内存地址

instance = memory;

看到上面指令重排的解释之后,那么我们来回顾一下未加volatile修饰符的单例为何会出现问题。假设有A、B两个线程去调用该单例方法,当A线程执行到single = new Single()时,如果编译器和处理器对指令重新排序,指令重排后:

//1:分配对象的内存空间

memory = allocate();

//3:设置instance指向刚分配的内存地址,此时对象还没被初始化

instance = memory;

//2:初始化对象

ctorInstance(memory);

当A线程执行到第二步(3:设置instance指向刚分配的内存地址,此时对象还没被初始化)变量single指向内存地址之后就不为null了,此时B线程进入第一个if,由于single已经不为null了,那么就不会执行到同步代码块,而是直接返回未初始化对象的变量single,从而导致后续代码报错。

解决方案

问题也搞清楚了,接下来我们就来看下如何解决这个问题。解决问题的关键就在于volatile关键字,原因就在于它的可见性:

写volatile修饰的变量时,JMM会把本地内存中值刷新到主内存 读

volatile修饰的变量时,JMM会设置本地内存无效

在多线程环境下,一个线程对共享变量的操作对其他线程是不可见的,Java提供了volatile来保证可见性,当一个变量被volatile修饰后,表示着线程本地内存无效,当一个线程修改共享变量后他会立即被更新到主内存中,其他线程读取共享变量时,会直接从主内存中读取。

当然,synchronize和Lock都可以保证可见性,synchronized和Lock能保证同一时刻只有一个线程获取锁然后执行同步代码,并且在释放锁之前会将对变量的修改刷新到主存当中。

更正后的单例

对比上面单例,下面单例在私有静态变量single前面加了修饰符volatile能够防止JVM指令重排,从而解决了single对象可能出现成员变量未初始化的问题。

publicclass Single {

private volatile staticSingle single;

private Single() {

}

publicstaticSingle getInstance() {

if(null== single) {

synchronized (Single.class) {

if(null== single) {

single = new Single();

}

}

}

returnsingle;

}

}

【编辑推荐】

【责任编辑:武晓燕 TEL:(010)68476606】

点赞 0

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
双重检查实现线程安全单例模式是一种常用的方式。在Java多线程编程中,双重检查定(DCL)单例模式可以确保只有一个实例被创建,并且在多线程环境下保持线程安全。\[1\] 该模式的实现方式是在getInstance()方法中进行双重检查,首先检查实例是否已经被创建,如果没有,则进入同步代码块。在同步代码块中,再次检查实例是否已经被创建,如果没有,则创建一个新的实例。这样可以避免多个线程同时创建实例的问题。 然而,需要注意的是,双重检查定模式在某些情况下可能会存在线程安全问题。具体来说,可能会出现代码指令重排序的情况,导致实例在多线程环境下不一致。\[2\] 为了解决这个问题,可以使用volatile关键字来修饰实例变量。volatile关键字可以确保变量的可见性和禁止指令重排序。通过在双重检查定模式中使用volatile关键字修饰实例变量,可以保证在多线程环境下实例的正确创建和访问。\[1\] 总结来说,双重检查定模式是一种常用的实现线程安全单例模式的方式。通过在getInstance()方法中进行双重检查,并使用volatile关键字修饰实例变量,可以确保在多线程环境下只有一个实例被创建,并且保持线程安全。\[1\]\[2\]\[3\] #### 引用[.reference_title] - *1* *2* [java单例模式的线程安全 JAVA多线程编程中的双重检查定(DCL单例(Double Check Lock))](https://blog.csdn.net/wuyuanshun/article/details/130101511)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] - *3* [【java单例模式双重检验](https://blog.csdn.net/qq_32088869/article/details/128027274)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值