多线程之volatile


前言

  volatile在Java中是一个变量修饰符,其作用是保证变量在线程之间的可见性以及防止指令重排序。

1 保证线程可见性

  关于线程之间的可见性,在之前的文章《并发编程之可见性》中已经详细描述过,所以这里不再赘述。

2 防止指令重排序

2.1 什么是指令重排序

  为了提升指令的执行效率,编译器会对代码结构进行重新排序,达到最佳效果。比如下面的代码:

int x = 1;
int y = 2;

经过编译器优化之后,其执行顺序可能会变成:

int y = 2;
int x = 1;

这种优化对于最终的结果来说并没有什么影响,保证了最终一致性。而现代处理器为了提升其指令执行的效率,采用了指令级并行技术(Instruction-LevelParallelism,ILP)来将多条指令重叠执行。如果不存在数据依赖性,处理器可以改变语句对应机器指令的执行顺序。

2.2 指令重排序可能造成的问题

  在单线程环境下,指令重排序并不会产生数据不一致的问题。但是在多线程的情况下,问题就出现了。下面借助双重检锁(DCL)单例模式来说明这个问题。

public class Singleton {
    private static Singleton instance;
    private int num;

    private Singleton() {
        num = 1;
    }

    public int getNum() {
        return num;
    }

    public static Singleton getInstance() {
        if (instance == null) {
            synchronized (Singleton.class) {
                if (instance == null) {
                    instance = new Singleton();
                }
            }
        }
        return instance;
    }
}

上面是双重检锁单例模式的代码,将其编译成字节码之后,instance = new Singleton()对应的字节码为:

17 new #4 <cn/spc/test/Singleton>
20 dup
21 invokespecial #5 <cn/spc/test/Singleton.<init>>
24 putstatic #3 <cn/spc/test/Singleton.instance>
27 aload_0

从上面的字节码可以看出,instance = new Singleton()主要有三个步骤:

  1. new #4 <cn/spc/test/Singleton>为对象分配空间,此时成员变量num的值为0。
  2. invokespecial #5 <cn/spc/test/Singleton.>调用构造方法,此时成员变量num的值为1。
  3. aload_0将对象的地址赋值给instance变量。

  在上面的过程中,如果步骤2和步骤3发生了指令重排序,问题就出现了。当第一个线程尝试获取Singleton实例,因为instance是null,所以会创建Singleton实例。假设这个线程在创建Singleton实例执行时,先执行了步骤1,再执行了步骤3,最后执行了步骤2。对于该线程来说,它最终得到了一个正确的Singleton实例。但是,如果有另外一个线程,在第一个线程执行到步骤3的时候,尝试获取Singleton实例,因为这个时候已经将对象的地址赋值给了instance变量,所以这个线程也可以拿到instance。但是,这个时候第一个线程还没有执行步骤2,所以instance的num成员变量值为0而不是1,这意味着这个线程拿到了一个中间状态的Singleton实例。

2.3 使用volatile防止指令重排序

  为了解决上面的问题,只需要用volatile修饰成员变量instance即可。

public class Singleton {
    private static volatile Singleton instance;
    private int num;

    private Singleton() {
        num = 1;
    }

    public int getNum() {
        return num;
    }

    public static Singleton getInstance() {
        if (instance == null) {
            synchronized (Singleton.class) {
                if (instance == null) {
                    instance = new Singleton();
                }
            }
        }
        return instance;
    }
}

2.4 volatile防止指令重排序的原理

  volatile是通过内存屏障来禁止指令重排序的,内存屏障前后两条指令不能进行重排序。JVM内存屏障策略为:

  • 在每个volatile写操作的前面插入一个StoreStore屏障。
  • 在每个volatile写操作的后面插入一个StoreLoad屏障。
  • 在每个volatile读操作的后面插入一个LoadLoad屏障。
  • 在每个volatile读操作的后面插入一个LoadStore屏障。

总结

  以上就volatile关于禁止指令重排序的内容,这里只是简单地描述了指令重排序的概念、指令重排序可能造成的问题、解决方法以及volatile禁止重排序的原理。关于指令重排序,其实还有很多更加深入的内容,后续有机会将会继续讲解。

  • 3
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
关键字的作用是保证多线程环境下共享变量的可见性和有序性。在多线程环境下,每个线程都有自己的工作内存,当线程要访问共享变量时,会先从主内存中加载变量到自己的工作内存中,然后进行操作。而使用volatile关键字修饰的变量,会强制线程在每次访问变量时都从主内存中读取最新的值,并将修改后的值立即写回主内存,从而确保了所有线程对该变量的可见性。同时,volatile关键字还可以防止重排序,保证了操作的有序性。引用引用了一篇详细介绍Java多线程volatile关键字的文章,通过示例代码对volatile关键字的使用进行了详细解释。引用引用了一段代码示例,展示了在多线程环境下使用volatile关键字的情况,以及当不使用volatile关键字时可能出现的问题。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [Java多线程 volatile关键字详解](https://download.csdn.net/download/weixin_38654220/12746815)[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^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] - *2* *3* [多线程(三) -- 内存(二) -- Volatile详解](https://blog.csdn.net/weixin_39724194/article/details/107385004)[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^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值