volatile

文章目录

volatile

可见性

问题:t 线程会从自己的高速缓存中读取run值,而无法获取到更新后的值,导致程序无法结束。

java 复制代码
@Slf4j
public class Test05 {
    static boolean run = true;
    public static void main(String[] args) {
        new Thread(() ->{
            int i=0;
            while(run){
                i++;
            }
        }).start();
        log.info("开始任务");
        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            throw new RuntimeException(e);
        }

        run = false;
        log.info("停止任务");
    }
}

解决方法:给run加上volatile(易变的)属性,避免从工作缓存中读取值,而必须到主缓存中获取值,保证数据在不同线程中的可见性。

java 复制代码
static volatile boolean run = true;

指令重排

问题:r1的结果可能为0,因为actor2中的两行代码可能会发生指令重排,导致执行顺序颠倒。

java 复制代码
int num = 0;
boolean ready = false; 
// 线程1 执行此方法 
public void actor1(I_Result r) {
    if(ready) { 
        r.r1 = num + num;
    }else {
        r.r1 = 1;
    } 
}
// 线程2 执行此方法 
public void actor2(I_Result r) {
    num = 2;
    ready = true;
}

解决:给ready添加volatile关键字

java 复制代码
volatile boolean ready = false;

volatile原理

volatile的底层实现原理是内存屏障,Memory Barrier(Memory Fence)

  • 对 volatile 变量的写指令后会加入写屏障
  • 对 volatile 变量的读指令前会加入读屏障

而有了屏障之后,可见性和有序性就有了很好的解决。

可见性:

  • 写屏障(sfence)保证在该屏障之前的,对共享变量的改动,都同步到主存当中
  • 读屏障(lfence)保证在该屏障之后,对共享变量的读取,加载的是主存中新数据

有序性:

  • 写屏障会确保指令重排序时,不会将写屏障之前的代码排在写屏障之后
  • 读屏障会确保指令重排序时,不会将读屏障之后的代码排在读屏障之前

虽然volatile可以保证可见性和有序性,但它并不能保证复合操作(如自增、自减或检查后执行逻辑)的原子性。因此,volatile并不能替代同步机制(如synchronized)来控制对共享资源的并发访问。

相关推荐
掘根25 分钟前
【消息队列项目】客户端四大模块实现
开发语言·后端·ruby
疯狂的挖掘机6 小时前
记一次基于QT的图片操作处理优化思路(包括在图上放大缩小,截图,画线,取值等)
开发语言·数据库·qt
cnxy1886 小时前
围棋对弈Python程序开发完整指南:步骤4 - 提子逻辑和劫争规则实现
开发语言·python·机器学习
意趣新7 小时前
C 语言源文件从编写完成到最终生成可执行文件的完整、详细过程
c语言·开发语言
.鸣7 小时前
set和map
java·学习
ha_lydms7 小时前
5、Spark函数_s/t
java·大数据·python·spark·数据处理·maxcompute·spark 函数
李艺为8 小时前
根据apk包名动态修改Android品牌与型号
android·开发语言
黄河滴滴8 小时前
java系统变卡变慢的原因是什么?从oom的角度分析
java·开发语言
侠客行03178 小时前
Mybatis二级缓存实现详解
java·mybatis·源码阅读
老华带你飞8 小时前
农产品销售管理|基于java + vue农产品销售管理系统(源码+数据库+文档)
java·开发语言·前端·数据库·vue.js·spring boot·后端