volatile保证可见性但不保证原子性的深入分析

Java垃圾回收中,关于jvm运行时刻的内存分配,其中有一个内存区域是jvm虚拟机栈,每一个线程运行时都有一个线程栈;
线程栈保存了线程运行时变量值信息,当线程访问某一个对象的值的时候,首先通过对象的引用找到对应在堆内存中的变量的值,然后把堆内存中变量的具体值load到线程本地内存中,建立一个变量副本,之后线程就不再和对象在堆内存变量值有任何关系,而是直接修改副本变量的值;

在修改完之后的某个时刻(线程退出之前),自动把线程变量副本的值回写到对象在堆中变量,这样在堆中的对象的值就产生了变化,如下图所示:

在这里插入图片描述
read and load 从主存复制变量到当前工作内存
use and assign 执行代码,改变共享变量值
store and write 用工作内存数据刷新主存相关内容

但是这些操作并不是原子性操作,也就是在read and load之后,如果主内存count变量发生修改之后,线程工作内存中的值由于已经加载,不会产生对应的变化,所以计算出的值会和预期不一样

对volatile修饰的变量,JVM虚拟机只是保证从主内存中加载到线程工作内存的值是最新的(只保证read and load

线程不安全的原因
如果线程1 和线程2 正在进行read and load 操作,发现主内存中的count变量的值是5,那么,都会加载这个最新的值;

在线程1 对count进行修改之后,会write到主内存中,主内存的count变量就会变为6;

但是由于线程2已经进行了read and load操作,也会更新主内存count的值为6;

导致两个线程即使使用了volatile关键字修饰,还是会存在并发的情况

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值