volatile内存语义

4 篇文章 0 订阅
本文详细阐述了volatile关键字在Java中的内存语义,包括写和读的规则,以及其通过JMM实现的内存一致性。重点讲解了volatile如何防止编译器和处理器重排序,以及在多线程环境下保证共享变量的可见性和有序性的案例。
摘要由CSDN通过智能技术生成


volatile写的内存语义

  • 当写一个volatile变量时,JMM会把该线程对应的本地内存中的共享变量值刷新到主内存。

volatile读的内存语义:

  • 当读一个volatile变量时,JMM会把该线程对应的本地内存置为无效,线程接下来将从主内存中读取共享变量。

volatile内存语义的实现原理

JMM属于语言级的内存模型,它确保在不同的编译器和不同的处理器平台之上,通过禁止特定类型的编译器重排序和处理器重排序,为程序员提供一致的内存可见性保证。

volatile禁止重排序规则

为了实现volatile的内存语义,JMM会限制编译器重排序,JMM针对编译器制定了volatile重排序规则表。

volatile禁止重排序场景

  1. 当第二个操作是volatile写时,不管第一个操作是什么,都不能重排序。
  2. 当第一个操作是volatile读时,不管第二个操作是什么,都不能重排序。
  3. 当第一个操作是volatile写,第二个操作是volatile读时,不能重排序。

有序性案例分析

案例描述

假设我们有一个共享变量count,它被多个线程同时访问和修改。为了避免线程安全问题,我们可以使用volatile关键字来确保变量的可见性和有序性。

错误代码:

public class SharedCount {  
    private volatile int count = 0;  
  
    public void increment() {  
        count++;  
    }  
  
    public int getCount() {  
        return count;  
    }  
}

在上面的代码中,我们使用volatile关键字来修饰变量count。这确保了当一个线程修改了count的值后,其他线程能够立即看到更新后的值。此外,volatile关键字还确保了内存屏障的插入,以确保指令的重排序不会影响到变量的可见性。

如果我们没有使用volatile关键字修饰count变量,那么可能会遇到线程安全问题。例如,当一个线程修改了count的值后,其他线程可能仍然看到旧的值,因为它们可能缓存了该变量的副本。此外,由于没有内存屏障的插入,指令的重排序可能导致不可预期的结果。

如何纠正:

为了纠正上述错误代码,我们可以使用volatile关键字来修饰count变量。这样,当一个线程修改了count的值后,其他线程能够立即看到更新后的值,并且由于内存屏障的插入,指令的重排序不会影响到变量的可见性。

纠正后

public class SharedCount {  
    private volatile int count = 0;  // 使用volatile关键字修饰count变量  
  
    public void increment() {  
        count++;  // 每次修改count时都会强制从主内存中读取最新值,并刷新到本地缓存中  
    }  
  
    public int getCount() {  
        return count;  // 每次读取count时都会从主内存中读取最新值,而不是从本地缓存中读取旧值  
    }  
}

通过使用volatile关键字,我们可以确保count变量的可见性和有序性,从而避免线程安全问题。

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

小阳小朋友

随便吧

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值