前言

Volatile关键字的作用主要有如下两个:
1.线程的可见性:当一个线程修改一个共享变量时,另外一个线程能读到这个修改的值。
2. 顺序一致性:禁止指令重排序。

一、线程可见性

我们先通过一个例子来看看线程的可见性:

public class VolatileTest {boolean flag = true;public void updateFlag() {this.flag = false;System.out.println("修改flag值为:" + this.flag);}public static void main(String[] args) {VolatileTest test = new VolatileTest();new Thread(() -> {while (test.flag) {}System.out.println(Thread.currentThread().getName() + "结束");}, "Thread1").start();new Thread(() -> {try {Thread.sleep(2000);test.updateFlag();} catch (InterruptedException e) {}}, "Thread2").start();}
}

打印结果如下,我们可以看到虽然线程Thread2已经把flag 修改为false了,但是线程Thread1没有读取到flag修改后的值,线程一直在运行

修改flag值为:false

我们把flag 变量加上volatile:

 volatile  boolean flag = true;

重新运行程序,打印结果如下。Thread1结束,说明Thread1读取到了flage修改后的值

修改flag值为:false
Thread1结束

说到可见性,我们需要先了解一下Java内存模型,Java内存模型如下所示:

线程之间的共享变量存储在主内存中(Main Memory)中,每个线程都一个都有一个私有的本地内存(Local Memory),本地内存中存储了该线程以读/写共享变量的副本。

所以当一个线程把主内存中的共享变量读取到自己的本地内存中,然后做了更新。在还没有把共享变量刷新的主内存的时候,另外一个线程是看不到的。

如何把修改后的值刷新到主内存中的?

现代的处理器使用写缓冲区临时保存向内存写入的数据。写缓冲区可以保证指令流水线持续运行,它可以避免由于处理器停顿下来等向内存写入数据而产生的延迟。同时,通过以批处理的方式刷新写缓冲区,以及合并写缓冲区中对同一内存地址的多次写,较少对内存总线的占用。但是什么时候写入到内存是不知道的。

所以就引入了volatile,volatile是如何保证可见性的呢?
在X86处理器下通过工具获取JIT编译器生成的汇编指令来查看对volatile进行写操作时,会多出lock addl。Lock前缀的指令在多核处理器下会引发两件事情:

将当前处理器缓存行的数据写回到系统内存。
这个写回内存的操作会使其他cpu里缓存了该内存地址的数据无效。
如果声明了volatile的变量进行写操作,JVM就会向处理器发送一条Lock前缀的指令,将这个变量所在缓存行的数据写回到系统内存。但是,就算写回到内存,如果其他处理器缓存的还是旧的,在执行操作就会有问题。所以,在多处理器下,为了保证各个处理器的缓存是一致的,就会实现缓存一致性协议,每个处理器通过嗅探在总线传播的数据来检查自己缓存的值是不是过期了,当处理器发现自己缓存行对应的内存地址被修改,就会将当前处理器的缓存行设置成无效状态,当处理器对这个数据进行修改操作的时候,会重新从系统内存中把数据读到处理器缓存里。

二、顺序一致性

在执行程序时,为了提高性能,编译器和处理器常常会对指令做重排序。重排序分为如下三种:

1属于编译器重排序,2和3属于处理器重排序。这些重排序可能会导致多线程程序出现内存可见性问题。
当变量声明为volatile时,Java编译器在生成指令序列时,会插入内存屏障指令。通过内存屏障指令来禁止重排序。
JMM内存屏障插入策略如下:
在每个volatile写操作的前面插入一个StoreStore屏障,后面插入一个StoreLoad屏障。
在每个volatile读操作后面插入一个LoadLoad,LoadStore屏障。

Volatile写插入内存屏障后生成指令序列示意图:

Volatile读插入内存屏障后生成指令序列示意图:

通过上面这些我们可以得出如下结论:编译器不会对volatile读与volatile读后面的任意内存操作重排序;编译器不会对volatile写与volatile写前面的任意内存操作重排序。

防止重排序使用案例:

public class SafeDoubleCheckedLocking {private volatile static Instance instane;public  static Instance getInstane(){if(instane==null){synchronized (SafeDoubleCheckedLocking.class){if(instane==null){instane=new Instance();}}}return instane;}
}

创建一个对象主要分为如下三步:

分配对象的内存空间。
初始化对象。
设置instance指向内存空间。

如果instane 不加volatile,上面的2,3可能会发生重排序。假设A,B两个线程同时获取,A线程获取到了锁,发生了指令重排序,先设置了instance指向内存空间。这个时候B线程也来获取,instance不为空,这样B拿到了没有初始化完成的单例对象(如下图)

二、Volatile与Synchronized比较

Volatile是轻量级的synchronized,因为它不会引起上下文的切换和调度,所以Volatile性能更好。
Volatile只能修饰变量,synchronized可以修饰方法,静态方法,代码块。
Volatile对任意单个变量的读/写具有原子性,但是类似于i++这种复合操作不具有原子性。而锁的互斥执行的特性可以确保对整个临界区代码执行具有原子性。
多线程访问volatile不会发生阻塞,而synchronized会发生阻塞。
volatile是变量在多线程之间的可见性,synchronize是多线程之间访问资源的同步性。

一文搞懂Volatile关键字的作用相关推荐

  1. 面试:一文搞懂 final 关键字的作用

    前言 Java语言支持的变量类型有: 类变量:独立于方法之外的变量,用 static 修饰. 实例变量:独立于方法之外的变量,不过没有 static 修饰. 局部变量:类的方法中的变量. 实例代码: ...

  2. 一文搞懂transient关键字

    一文搞懂transient关键字 1.序列化 1.1 什么是序列化 1.2 以文件存储为例实现序列化 2.transient关键字 2.1 添加transient关键字 2.2 静态变量不需要加tra ...

  3. java:彻底搞懂volatile关键字

    对于volatile这个关键字,相信很多朋友都听说过,甚至使用过,这个关键字虽然字面上理解起来比较简单,但是要用好起来却不是一件容易的事. 这篇文章将从多个方面来讲解volatile,让你对它更加理解 ...

  4. 20200428 线程安全(上)--彻底搞懂volatile关键字

    计算机在处理数据的过程中为什么会出现线程不安全的问题. 计算机在执行程序时,每条指令都是在CPU中执行的,而执行指令过程中会涉及到数据的读取和写入.由于程序运行过程中的临时数据是存放在主存(物理内存) ...

  5. 26张图带你彻底搞懂volatile关键字

    引子 小艾吃饭路上碰上小牛,忙问:你昨天面大厂面的咋样了?听说他们最喜欢问多线程相关知识. 小牛说:对啊,第一个问题我就讲了20分钟,直接把面试官讲服了. 小艾忙问:什么问题能讲这么久?是不是问你情感 ...

  6. 一文读懂 volatile 关键字

    点击上方 好好学java ,选择 星标 公众号 重磅资讯.干货,第一时间送达 今日推荐:有了这 4 款工具,老板再也不怕我写烂SQL了个人原创+1博客:点击前往,查看更多 作者:对弈 来源:https ...

  7. 万字长文详细搞懂 volatile 关键字

    volatile 这个关键字大家都不陌生,这个关键字一般通常用于并发编程中,是 Java 虚拟机提供的轻量化同步机制,你可能知道 volatile 是干啥的,但是你未必能够清晰明了的知道 volati ...

  8. 网络知识扫盲,一文搞懂 DNS

    在找工作面试的过程中,面试官非常喜欢考察基础知识,除了数据结构与算法之外,网络知识也是一个非常重要的考察对象. 而网络知识,通常是很抽象,不容易理解的,有很多同学就在这里裁了跟头.为了更好地通过面试, ...

  9. ES6学习——一文搞懂ES6

    ES6学习--一文搞懂ES6 es6介绍 ES全称EcmaScript,是脚本语言的规范,而平时经常编写的EcmaScript的一种实现,所以ES新特性其实就是指JavaScript的新特性. 为什么 ...

最新文章

  1. 开心网外挂开发之 XML序列化于反序列化
  2. BZOJ 2716: [Violet 3]天使玩偶
  3. 2020-11-19(栈帧)
  4. vue不是内部或外部命令
  5. deprecated_@Deprecated新外观可能是什么?
  6. Android 蓝牙驱动专题分析(2)--- 蓝牙驱动代码流程、kernel dump、tombstone问题分析
  7. 首款基于龙芯的域名系统服务器发布,首款基于龙芯CPU的国产域名服务器发布
  8. NOIP2012 国王游戏(贪心)
  9. 地图标识符号大全_起名字大全男孩 男孩名字,起名字大全男孩
  10. sigmoid和softmax
  11. 策略模式(用策略模式实现我们淘宝,京东,美团等等简易满减活动)
  12. win10 安装并跑通 mmdetection
  13. 数据结构之B树、B+树、B*树
  14. 爸爸妈妈儿子女儿吃水果问题以及五个哲学家吃饭问题
  15. Linux fsck命令详解
  16. 架构师速成5.2-价值观和目标
  17. DM8数据库备份与恢复 DMRMAN
  18. error TS7056
  19. 解密中国互联网构成与发展
  20. Linux如何识别U盘

热门文章

  1. VMware 11 安装苹果系统
  2. python爬虫解决频繁访问_爬虫遇到IP访问频率限制的解决方案
  3. 这所计算机实力顶尖的大学,也成立人工智能学院
  4. cocos2dx手游开发之环境搭建
  5. 数据分析行业出路太窄?那是你根本不懂什么是数据分析
  6. typescript error TS2322: Type ‘Timeout‘ is not assignable to type ‘number‘.
  7. 2021-06-24Leetcode240.二维数组中的查找
  8. Existing lock /var/run/yum.pid: another copy is running as pid 22873. Another app is currently hold.
  9. dsp广告和信息流广告区别_信息流、DSP、联盟广告、SEM有什么区别?
  10. java poi word宏_全面了解POI操作Microsoft Office(Word、Excel、PowerPoint)