java 中的volatile关键字

 更新时间:2021年12月12日 12:03:49   作者:bkpp976  
这篇文章主要介绍了java 中的volatile关键字,volatile在多处理器开发中保证共享变量的“可见性”。可见性的意思是当一个线程修改一个共享变量时,另一个一个线程立马可以读到这个修改的值。下面我们来看看文章的具体介绍内容吧

1.volatile实现可见性的原理是什么?

volatile变量修饰的共享变量进行写操作的时候汇编代码会多出一个Lock前缀指令。

在该指令下,多核处理器会引发两件事:

  • 将当前处理器缓存行的数据写回系统内存
  • 这个写回内存的操作会使在其他CPU里缓存了该内存地址的数据无效

这里需要简单了解CPU缓存一致性问题:多核处理器环境下,每个CPU都有自己的缓存行,缓存了内存中的数据,要维护多个CPU中缓存的数据一致性,就需要解决两个问题:

  • 一是写传播(某个CPU里的cache数据更新时,需要传播到其他CPU的cache中);
  • 二是事务的串行化执行(在某个CPU里对数据的修改,在其他CPU中看起来顺序是一样的,也就是要引入近似[锁]的概念,保证同一时刻只有一个CPU可以对数据做修改);

写传播是通过[总线嗅探]完成的:通过总线把修改数据的事件广播通知给其他所有的核心,每个CPU核心都会监听总线上的广播事件,并检查是否有相同的数据在自己的Cache里面;而事务的串行化则通过[MESI协议]来完成。

MESI(Modified(已修改)、Exclusive(独占)、Shared(共享)、Ivalidated(已失效))协议中,如果要修改一个共享数据,不能直接修改,要先向其他CPU广播一个请求,把其他CPU cache中对应的数据状态改为Invalidated;以后其他CPU在读取标记为Invalidated的数据时,需要强制从内存中读取数据。

2.演示volatile的可见性

public class VolatileDemo {
    static  int flag = 1;  // 定义一个共享变量
    public static void main(String[] args) {
        // 两个线程,一个线程负责读取flag的值,另一个线程负责修改flag的值
        new Thread(){
            int localflag = flag;
            @Override
            public void run() {
                while(true){
                    //flag被修改后就跟localflag不一样了
                    if(localflag!=flag){
                        System.out.println("读到了flag修改后的值:"+ flag);
                        //把读到的值赋值给本地变量
                        localflag = flag;
                    }
                }
            }
        }.start();

        new Thread(){
            int localflag = flag;
            @Override
            public void run() {
                while (true){
                    //一直对flag的值进行修改
                    System.out.println("对flag的值进行修改:"+ ++localflag);
                    flag = localflag;
                    //休眠一秒更好地观察结果
                    try {
                        Thread.sleep(1000);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }
        }.start();


    }
}

可以看到另一个线程并不能及时读取到被修改的值。

共享变量用volatile修饰后:

public class VolatileDemo {
    static  volatile int flag = 1;
    public static void main(String[] args) {
        // 两个线程,一个线程负责读取flag的值,另一个线程负责修改flag的值
        new Thread(){
            int localflag = flag;
            @Override
            public void run() {
                while(true){
                    //flag被修改后就跟localflag不一样了
                    if(localflag!=flag){
                        System.out.println("读到了flag修改后的值:"+ flag);
                        //把读到的值赋值给本地变量
                        localflag = flag;
                    }
                }
            }
        }.start();

        new Thread(){
            int localflag = flag;
            @Override
            public void run() {
                while (true){
                    //一直对flag的值进行修改
                    System.out.println("对flag的值进行修改:"+ ++localflag);
                    flag = localflag;
                    //休眠一秒更好地观察结果
                    try {
                        Thread.sleep(1000);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }
        }.start();


    }
}

可以看到用volatile修饰后,每次另一个线程总能读取到修改后的值。

到此这篇关于java 中的volatile关键字的文章就介绍到这了,更多相关volatile关键字内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Spring Bean名称不会被代理的命名技巧

    Spring Bean名称不会被代理的命名技巧

    Spring Bean一些使用小细节就是在不断的源码探索中逐步发现的,今天就来和小伙伴们聊一下通过 beanName 的设置,可以让一个 bean 拒绝被代理
    2023-11-11
  • MySQL中关键字UNION和UNION ALL的区别

    MySQL中关键字UNION和UNION ALL的区别

    本文主要介绍了MySQL中关键字UNION和UNION ALL的区别,深入探讨UNION和UNION ALL的定义、用法、主要区别,具有一定的参考价值,感兴趣的可以了解一下
    2024-06-06
  • 快速入门介绍Java中强大的String.format()

    快速入门介绍Java中强大的String.format()

    这篇文章主要给大家介绍了如何快速入门介绍Java中强大的String.format()的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧。
    2018-03-03
  • Java基础教程之数组的定义与使用

    Java基础教程之数组的定义与使用

    Java语言的数组是一个由固定长度的特定类型元素组成的集合,它们的数据类型必须相同,声明变量的时候,必须要指定参数类型,这篇文章主要给大家介绍了关于Java基础教程之数组的定义与使用的相关资料,需要的朋友可以参考下
    2021-09-09
  • idea右键没有java class选项问题解决方案

    idea右键没有java class选项问题解决方案

    这篇文章主要介绍了idea右键没有java class选项问题解决方案,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-04-04
  • SpringBoot解析指定Yaml配置文件的实现过程

    SpringBoot解析指定Yaml配置文件的实现过程

    这篇文章主要介绍了SpringBoot解析指定Yaml配置文件,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2023-03-03
  • Java添加Word文本水印和图片水印

    Java添加Word文本水印和图片水印

    这篇文章主要介绍了Java添加Word文本水印和图片水印,文章图文讲解的很清晰,有对于这方面不懂得同学可以学习下
    2021-02-02
  • springboot+mybaties项目中扫描不到@mapper注解的解决方法

    springboot+mybaties项目中扫描不到@mapper注解的解决方法

    本文主要介绍了springboot+mybaties项目中扫描不到@mapper注解的解决方法,该报错表明扫描不到Mapper层,具有一定的参考价值,感兴趣的可以了解一下
    2024-05-05
  • Java大数字运算之BigInteger

    Java大数字运算之BigInteger

    在Java中提供了大数字的操作类,即 java.math.BigInteger 类与 java.math.BigDecimal 类。其中,BigInteger 类是针对大整数的处理类,这里有Integer 类的解释,使用方法和实例,需要的朋友可以参考下。
    2017-08-08
  • Jax-rs规范REST接口文件上传代码实例

    Jax-rs规范REST接口文件上传代码实例

    这篇文章主要介绍了Jax-rs规范REST接口文件上传代码实例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-09-09

最新评论