volatile原理

volatile简介

Java内存模型告诉我们,各个线程会将共享变量从主内存中拷贝到工作内存,然后执行引擎会基于工作内存中的数据进行操作处理。 线程在工作内存进行操作后何时会写到主内存中? 这个时机对普通变量是没有规定的,而针对volatile修饰的变量给Java 虚拟机特殊的约定,线程对 volatile变量的修改会立刻被其他线程所感知,即不会出现数据脏读的现象,从而保证数据的“可见性”。

一言以蔽之,被volatile修饰的变量能够保证每个线程能够获取该变量的最新值,从而避免出现数据脏读的现象

volatile实现原理

volatile是怎样实现了?比如一个很简单的Java代码:

instance = new Instancce() //instance是volatile变量

在生成汇编代码时会在volatile修饰的共享变量进行写操作的时候会多出Lock前缀的指令。 我们想这个Lock指令肯定有神奇的地方,那么Lock前缀的指令在多核处理器下会发现什么事情了?主要有这两个方面的影响:

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

为了提高处理速度,处理器不直接和内存进行通信,而是先将系统内存的数据读到内部缓存(L1,L2或其他)后再进行操作,但操作完不知道何时会写到内存。 如果对声明了volatile的变量进行写操作,JVM就会向处理器发送一条Lock前缀的指令,将这个变量所在缓存行的数据写回到系统内存。但是,就算写回到内存,如果其他处理器缓存的值还是旧的,再执行计算操作就会有问题。

在多处理器下,为了保证各个处理器的缓存是一致的,就会实现缓存一致性协议,每个处理器通过嗅探在总线上传播的数据来检查自己缓存的值是不是过期了,当处理器发现自己缓存行对应的内存地址被修改,就会将当前处理器的缓存行设置成无效状态,当处理器对这个数据进行修改操作的时候,会重新从系统内存中把数据读到处理器缓存里。 因此,经过分析我们可以得出如下结论:

  • Lock前缀的指令会引起处理器缓存写回内存
  • 一个处理器的缓存回写到内存会导致其他工作内存中的缓存失效
  • 当处理器发现本地缓存失效后,就会从主内存中重读该变量数据,即可以获取当前最新值

这样volatile变量通过这样的机制就使得每个线程都能获得该变量的最新值。

volatile的happens-before关系

happens-before中的volatile 变量规则(Volatile Variable Rule):对一个 volatile 变量的写操作先行发生于后面对这个变量的读操作。

public class VolatileExample {private int a = 0;private volatile boolean flag = false;public void writer(){a = 1;          //1flag = true;   //2}public void reader(){if(flag){      //3int i = a; //4}}
}

对应的happens-before关系如下:

加锁线程A先执行writer方法,然后线程B执行reader方法。 图中每一个箭头两个节点就代码一个happens-before关系:

  • 黑色的代表根据程序顺序规则推导出来
  • 红色的是根据volatile变量的写happens-before 于任意后续对volatile变量的读
  • 蓝色的就是根据传递性规则推导出来的 这里的2 happen-before 3,同样根据happens-before规则定义: 如果A happens-before B,则A的执行结果对B可见,并且A的执行顺序先于B的执行顺序, 我们可以知道操作2执行结果对操作3来说是可见的,也就是说当线程A将volatile变量 flag更改为true后线程B就能够迅速感知。

volatile的内存语义

public class VolatileExample {private int a = 0;private volatile boolean flag = false;public void writer(){a = 1;          //1flag = true;   //2}public void reader(){if(flag){      //3int i = a; //4}}
}

假设线程A先执行writer方法,线程B随后执行reader方法,初始时线程的本地内存中flag和a都是初始状态,下图是线程A执行volatile写后的状态图:

当volatile变量写后,线程B中本地内存中共享变量就会置为失效的状态,因此线程B需要从主内存中去读取该变量的最新值。下图就展示了线程B读取同一个volatile变量的内存变化示意图:

从横向来看,线程A和线程B之间进行了一次通信,线程A在写volatile变量时,实际上就像是给B发送了一个消息告诉线程B你现在的值都是旧的了,然后线程B读这个volatile变量时就像是接收了线程A刚刚发送的消息。既然是旧的了,那线程B该怎么办了?自然而然就只能去主内存去取啦。

volatile的内存语义实现

为了性能优化,JMM在不改变正确语义的前提下,会允许编译器和处理器对指令序列进行重排序,那如果想阻止重排序要怎么办了? 答案是可以添加内存屏障

四类JMM内存屏障:

Java编译器会在生成指令系列时在适当的位置会插入内存屏障指令来禁止特定类型的处理器重排序。 为了实现volatile的内存语义,JMM会限制特定类型的编译器和处理器重排序,JMM会针对编译器制定volatile重排序规则表:

"NO"表示禁止重排序。 为了实现volatile内存语义时,编译器在生成字节码时,会在指令序列中插入内存屏障来禁止特定类型的处理器重排序。 对于编译器来说,发现一个最优布置来最小化插入屏障的总数几乎是不可能的,为此,JMM采取了保守策略:

  • 在每个volatile写操作的前面插入一个StoreStore屏障
  • 在每个volatile写操作的后面插入一个StoreLoad屏障

  • 在每个volatile读操作的后面插入一个LoadLoad屏障
  • 在每个volatile读操作的后面插入一个LoadStore屏障

需要注意的是:volatile写操作是在前面和后面分别插入内存屏障,而volatile读操作是在后面插入两个内存屏障。

volatile和synchronized的区别

  1. volatile本质是告诉JVM当前变量在寄存器(工作内存)中是无效的,需要去主内存重新读取;synchronized是锁定当前变量,只有持有锁的线程才可以访问该变量,其他线程都被阻塞直到该线程的变量操作完成;
  2. volatile仅仅能使用在变量级别;synchronized则可以使用在变量、方法和类级别;
  3. volatile仅仅能实现变量修改的可见性,不能保证原子性;而synchronized则可以保证变量修改的可见性和原子性;
  4. volatile不会造成线程的阻塞;synchronized可能会造成线程的阻塞;
  5. volatile修饰的变量不会被编译器优化;synchronized修饰的变量可以被编译器优化。

免费Java高级资料需要自己领取,涵盖了Java、Redis、MongoDB、MySQL、Zookeeper、Spring Cloud、Dubbo高并发分布式等教程,一共30G。

传送门:https://mp.weixin.qq.com/s/JzddfH-7yNudmkjT0IRL8Q

转载于:https://www.cnblogs.com/yuxiang1/p/10894222.html

Java并发编程,3分分钟深入分析volatile的实现原理相关推荐

  1. Java并发编程:JMM和volatile关键字

    Java内存模型 随着计算机的CPU的飞速发展,CPU的运算能力已经远远超出了从主内存(运行内存)中读取的数据的能力,为了解决这个问题,CPU厂商设计出了CPU内置高速缓存区.高速缓存区的加入使得CP ...

  2. Java并发编程(三)volatile域

    相关文章 Java并发编程(一)线程定义.状态和属性 Java并发编程(二)同步 Android多线程(一)线程池 Android多线程(二)AsyncTask源代码分析 前言 有时仅仅为了读写一个或 ...

  3. 进阶笔记——java并发编程三特性与volatile

    欢迎关注专栏:Java架构技术进阶.里面有大量batj面试题集锦,还有各种技术分享,如有好文章也欢迎投稿哦.微信公众号:慕容千语的架构笔记.欢迎关注一起进步. 前言 前面讲过使用synchronize ...

  4. 聊聊并发(一)——深入分析Volatile的实现原理

    引言 在多线程并发编程中synchronized和Volatile都扮演着重要的角色,Volatile是轻量级的synchronized,它在多处理器开发中保证了共享变量的"可见性" ...

  5. Java并发编程—无锁互斥机制及CAS原理

    目录 一.CAS简介 二.AtomicInteger代码演示 三.CAS 实现 四.弊端 一.CAS简介 在计算机科学中,比较和交换(Conmpare And Swap)是用于实现多线程同步的原子指令 ...

  6. 【Java并发编程:volatile关键字之解析】

    Java并发编程:volatile关键字解析 - Matrix海子 - 博客园 在Java 5之前,volatile是一个备受争议的关键字:因为在程序中使用它往往会导致出人意料的结果.在Java 5之 ...

  7. Java并发编程最佳实例详解系列

    Java并发编程最佳实例详解系列: Java并发编程(一)线程定义.状态和属性 Java并发编程(一)线程定义.状态和属性 线程是指程序在执行过程中,能够执行程序代码的一个执行单元.在java语言中, ...

  8. Java并发编程—常见面试题

    建议: 学习java并发前需要先掌握JVM知识 关于下面问题档案的详细解析都在后面推荐的相关系列文章中 一.线程安全相关 1.什么叫线程安全? 线程安全就是说多线程访问同一代码,不会产生不确定的结果. ...

  9. java面试-Java并发编程(六)——线程间的通信

    多条线程之间有时需要数据交互,下面介绍五种线程间数据交互的方式,他们的使用场景各有不同. 1. volatile.synchronized关键字 PS:关于volatile的详细介绍请移步至:Java ...

最新文章

  1. jquery学习开发资料
  2. PyCharm简单使用介绍及注意事项
  3. 移动端HTML5长按图片会选中页面中某个位置的文字
  4. Django报错SocialApp matching query does not exist以及Django的SITE_ID = 1的含义
  5. Linux的sort命令用法
  6. C++ Byte转十六进制字符串输出
  7. python获取类的类属性_Python中如何获取类属性的列表
  8. 列顺序对SQL Server复合索引的影响
  9. 跟我一起考PMP---项目成本管理
  10. 使用SQL语句获得服务器名称和IP 地址
  11. Autodesk Map 3D 2012 新功能介绍
  12. Linux进阶之日志管理
  13. logistic回归分析优点_糖尿病前期患者焦虑现状调查及影响因素分析
  14. Bootstrap的js插件之側边栏停靠(affix)
  15. background的使用方法
  16. sizeof和strlen使用详解
  17. 访问控制列表之基本ACL、高级ACL 、 高级ACL之ICMP、高级ACL之telnet
  18. java实现轮播图片_Banner框架实现图片轮播
  19. python中的max(x,key=str)
  20. python文件二进制读写_Python如何读写二进制文件

热门文章

  1. mysql查询中文_MySQL中like查询中文问题的解决
  2. spring aop搭建(2) :基于代码的实现
  3. 实战|简单绕过waf拿下赌博网站
  4. idea中的markdown文档如何插入图片
  5. Jquery中AJAX参数详细介绍
  6. ReactNative-WebView组件
  7. Linux 修改主机名 和 ip 映射关系
  8. Scrapy:Python的爬虫框架----原理介绍
  9. {'张三丰': 101, '无忌': 102, '赵敏': 102} (Python)
  10. WGAN-GP 学习笔记