通过示例了解挥发

未来的波动 我们已经花了几个月的时间来稳定Plumbr中的锁定检测功能 。 在此期间,我们遇到了许多棘手的并发问题。 许多问题是独特的,但是一种特殊类型的问题一直反复出现。

您可能已经猜到了–滥用volatile关键字。 我们已经发现并解决了许多问题,其中大量使用volatile使应用程序的任意部分变慢,延长了锁定保持时间,并最终使JVM屈服。 反之亦然-授予过于宽松的访问策略会引发一些令人讨厌的并发问题。

我想每个Java开发人员都会回想起该语言的第一步。 使用手册和教程的日子日复一日。 这些教程都有关键字列表,其中volatile是最可怕的关键字之一。 随着时间的流逝,越来越多的代码不需要使用此关键字,我们很多人都忘记了volatile的存在。 直到生产系统开始以无法预测的方式破坏数据或死亡。 调试这种情况迫使我们中的一些人真正理解了这个概念。 但是我敢打赌,这并不是一个令人愉快的课程,所以也许我可以通过一个简单的例子来阐明一些概念,从而节省一些时间。

挥发作用的例子

该示例模拟了一个银行办公室。 银行办公室的类型,您可以在此从售票机中选择队列编号,然后在您前面的队列得到处理后等待邀请。 为了模拟这样的办公室,我们创建了以下示例,该示例由两个线程组成。

这两个线程中的第一个被实现为CustomerInLine。 这是一个线程,除了等待NEXT_IN_LINE中的值与客户的票证匹配外,什么也不做。 票号被硬编码为#4。 时间到了( NEXT_IN_LINE> = 4),线程将宣布等待结束并结束。 这模拟了一个有一些客户在排队的到达办公室的客户。

排队实现在Queue类中,该类运行一个循环,该循环调用下一个客户,然后通过为每个客户休眠200ms来模拟与该客户的工作。 呼叫下一个客户后,存储在类变量NEXT_IN_LINE中的值将增加1。

public class Volatility {static int NEXT_IN_LINE = 0;public static void main(String[] args) throws Exception {new CustomerInLine().start();new Queue().start();}static class CustomerInLine extends Thread {@Overridepublic void run() {while (true) {if (NEXT_IN_LINE >= 4) {break;}}System.out.format("Great, finally #%d was called, now it is my turn\n",NEXT_IN_LINE);}}static class Queue extends Thread {@Overridepublic void run() {while (NEXT_IN_LINE < 11) {System.out.format("Calling for the customer #%d\n", NEXT_IN_LINE++);try {Thread.sleep(200);} catch (InterruptedException e) {e.printStackTrace();}}}}
}

因此,运行此简单程序时,您可能希望该程序的输出类似于以下内容:

Calling for the customer #1
Calling for the customer #2
Calling for the customer #3
Calling for the customer #4
Great, finally #4 was called, now it is my turn
Calling for the customer #5
Calling for the customer #6
Calling for the customer #7
Calling for the customer #8
Calling for the customer #9
Calling for the customer #10

看来,这个假设是错误的。 取而代之的是,您将看到通过10个客户的列表进行的队列处理,并且不幸的线程模拟了#4客户,从不对其看到邀请发出警报。 发生了什么,为什么客户仍然坐在那里无休止地等待呢?

分析结果

您在这里面临的是将JIT优化应用于代码,该代码将对NEXT_IN_LINE变量的访问进行缓存。 两个线程都有自己的本地副本,并且CustomerInLine线程从不看到Queue实际增加了线程的值。 如果现在您认为这是JVM中的一种可怕的错误,那么您就不完全正确了–允许编译器这样做以避免每次都重新读取该值。 因此,您可以提高性能,但要付出代价-如果其他线程更改状态,则缓存副本的线程将不知道该状态,并使用过时的值进行操作。

对于volatile正是这种情况。 使用此关键字,编译器将被警告特定状态是易失性的,并且每次执行循环时,代码都被强制重新读取该值。 有了这些知识,我们就可以进行简单的修复-只需将NEXT_IN_LINE的声明更改为以下内容,您的客户就不会永远坐在队列中:

static volatile int NEXT_IN_LINE = 0;

对于那些只了解volatile用例而感到满意的人,您将很高兴。 只是要注意附加的成本–当您开始声明所有内容都是易失性时,您正在迫使CPU忽略本地缓存并直接进入主内存,从而减慢了代码的速度并阻塞了内存总线。

引擎盖下易挥发

对于那些希望详细了解该问题的人,请和我在一起。 要查看底层发生了什么,请打开调试以查看JIT从字节码生成的汇编代码。 通过指定以下JVM选项可以实现此目的:

-XX:+UnlockDiagnosticVMOptions -XX:+PrintAssembly

在启用和禁用volatile的情况下启用这些选项的情况下运行程序,可以为我们提供以下重要见解:

运行不带volatile关键字的代码,表明我们在指令0x00000001085c1c5a上可以比较两个值。 当比较失败时,我们继续从0x00000001085c1c60到0x00000001085c1c66,后者跳回到0x00000001085c1c60,并产生无限循环。

0x00000001085c1c56: mov    0x70(%r10),%r11d0x00000001085c1c5a: cmp    $0x4,%r11d0x00000001085c1c5e: jge    0x00000001085c1c68  ; OopMap{off=64};*if_icmplt; - Volatility$CustomerInLine::run@4 (line 14)0x00000001085c1c60: test   %eax,-0x1c6ac66(%rip)        # 0x0000000106957000;*if_icmplt; - Volatility$CustomerInLine::run@4 (line 14);   {poll}0x00000001085c1c66: jmp    0x00000001085c1c60  ;*getstatic NEXT_IN_LINE; - Volatility$CustomerInLine::run@0 (line 14)0x00000001085c1c68: mov    $0xffffff86,%esi

使用volatile关键字后,我们可以看到在指令0x000000010a5c1c40上我们将值加载到寄存器中,在0x000000010a5c1c4a上将其与保护值4进行了比较。如果比较失败,我们从0x000000010a5c1c4e跳回到0x000000010a5c1c40,再次为新的值加载值校验。 这样可以确保我们看到NEXT_IN_LINE变量的值已更改

0x000000010a5c1c36: data32 nopw 0x0(%rax,%rax,1)0x000000010a5c1c40: mov    0x70(%r10),%r8d    ; OopMap{r10=Oop off=68};*if_icmplt; - Volatility$CustomerInLine::run@4 (line 14)0x000000010a5c1c44: test   %eax,-0x1c1cc4a(%rip)        # 0x00000001089a5000;   {poll}0x000000010a5c1c4a: cmp    $0x4,%r8d0x000000010a5c1c4e: jl     0x000000010a5c1c40  ;*if_icmplt; - Volatility$CustomerInLine::run@4 (line 14)0x000000010a5c1c50: mov    $0x15,%esi

现在,希望这种解释可以使您摆脱几个讨厌的错误。

翻译自: https://www.javacodegeeks.com/2014/08/understanding-volatile-via-example.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/363005.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

HDU 1212 Big Number

题意&#xff1a;给一数字字符串s ( ns.size()<1000 ) 和数字m (<1e5) 求s%m 模拟除法&#xff0c; k初值0&#xff0c;按s[0]...累乘相加&#xff0c;把字符串还原成数字&#xff0c;比m大时-m&#xff0c;继续按位还原到s[n-1] 此时剩下的k再%m即为所求 #include<…

BOM之navigator对象和用户代理检测

前面的话 navigator对象现在已经成为识别客户端浏览器的事实标准&#xff0c;navigator对象是所有支持javascript的浏览器所共有的。本文将详细介绍navigator对象和用户代理检测 属性 与其他BOM对象的情况一样&#xff0c;每个浏览器中的navigator对象也都有一套自己的属性。下…

智能自动PPR更改事件策略

ADF开发人员普遍认为&#xff0c;将迭代器绑定更改事件策略设置为ppr在性能方面不是一件好事&#xff0c;因为此策略会强制框架刷新每个请求上绑定到此迭代器的所有属性绑定。 这不是真的&#xff01; 框架仅刷新在请求期间已更改的属性和依赖于已更改属性的属性。 让我们考虑…

装饰器设计模式的应用

嗨&#xff0c;您好&#xff01; 今天&#xff0c;我将展示装饰设计模式的实际应用。 装饰器设计模式是一种广泛使用的设计模式&#xff0c;同时在运行期间处理图形&#xff0c;树木和动态更改。 如果您正在寻找或尝试进行递归&#xff0c;这也是一个不错的选择。 我喜欢它。…

构造函数 基本使用

相关知识点&#xff1a; 构造函数、原型对象、实例对象 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><meta h…

Vista,Windows7中给IIS7添加PHP支持

截止到发文时&#xff08;2009年9月25日&#xff09;&#xff0c;PHP是最新版本为5.3.0&#xff0c;但是5.3.0在IIS中的运行方式是FastCGI&#xff0c;要在Vista的IIS7上实现这一点我始终没弄成。最后我还是用的老方法&#xff0c;ISAPI&#xff0c;选用了一个5.3以前的版本&am…

python画手绘图

第一步&#xff1a;插入代码 #e17.1HandDrawPic.py from PIL import Image import numpy as np vec_el np.pi/2.2 # 光源的俯视角度&#xff0c;弧度值 vec_az np.pi/4. # 光源的方位角度&#xff0c;弧度值 depth 10. # (0-100) im Image.open(C:\\Users\\Thinkpad\\Deskt…

解读阿里巴巴集团的“大中台、小前台”组织战略

解读阿里巴巴集团的“大中台、小前台”组织战略 https://www.iyiou.com/p/92012.html 亿欧导读 ] 阿里的“中台战略” 不是一个简单的组织变革&#xff0c;还有业务变革、机制变革、技术架构变革的一次全面转型。 【编者按】阿里巴巴“大中台小前台”的中台战略的官方提法源自2…

深入理解DOM节点关系

前面的话 DOM可以将任何HTML描绘成一个由多层节点构成的结构。节点分为12种不同类型&#xff0c;每种类型分别表示文档中不同的信息及标记。每个节点都拥有各自的特点、数据和方法&#xff0c;也与其他节点存在某种关系。节点之间的关系构成了层次&#xff0c;而所有页面标记则…

分布式锁(基于redis和zookeeper)详解

分布式锁&#xff08;基于redis和zookeeper&#xff09;详解 https://blog.csdn.net/a15835774652/article/details/81775044 为什么写这篇文章&#xff1f; 目前网上大部分的基于zookeeper&#xff0c;和redis的分布式锁的文章都不够全面。要么就是特意避开集群的情况&#xf…

JavaFX技巧12:在CSS中定义图标

当您是像我这样来自Swing的UI开发人员时&#xff0c;您很有可能仍在代码中直接设置图像/图标。 最可能是这样的&#xff1a; import javafx.scene.control.Label; import javafx.scene.image.ImageView;public class MyLabel extends Label {public MyLabel() {setGraphic(new…

第八次点评

本周心得&#xff1a; 本周没有作业 &#xff0c;批改的上周的作业。需求分析以及团队的明确分工是开发前准备的要素。本次批改主要侧重于需求分析的调研。 博客园地址https://www.cnblogs.com/zhaojh123/ 博客园点评博客&#xff1a; https://www.cnblogs.com/yanqignkui-123/…

JDBC、Tomcat为什么要破坏双亲委派模型?

问题一&#xff1a;双亲委派模型是什么 如果一个类加载器收到了加载某个类的请求&#xff0c;则该类加载器并不会去加载该类&#xff0c;而是把这个请求委派给父类加载器&#xff0c;每一个层次的类加载器都是如此&#xff0c;因此所有的类加载请求最终都会传送到顶端的启动类加…

dpdk之路-环境部署

dpdk实验环境部署 1、实验环境说明 vmware workstatioin 12 centos 7.5.1804 dpdk-stable-18.11.1 2、实验步骤 &#xff08;1&#xff09;虚拟机安装 http://vault.centos.org/7.5.1804/isos/x86_64/从链接下载CentOS-7-x86_64-DVD-1804.iso&#xff0c;安装时需要准备3张虚拟…

基于知识图谱的医疗诊断系统论文

本作品禁止任何人/企业申请专利&#xff0c;禁止任何人使用本作品参加任何比赛或作为毕业设计&#xff0c;如使用本作品源码进行商业用途务必联系作者。 一.科学性 1.研究意义 信息科技经过 60 余年的发展&#xff0c;已经普及到社会生活的每一个角落。随着信息技术在国家治理、…

华为云(ECS)-linux服务器中-Ubuntu图形界面安装-解决root登录受限-VNCviwer/Teamviwer远程访问教程...

安装ubuntu-desktop 1.更新软件库 apt-get update2.升级软件 apt-get upgrade3.安装桌面 apt-get install ubuntu-desktop 解决root登录受限 华为云登录进去是guest用户&#xff0c;不能选择登录用户且不需要密码就即可登录。 登录进去会如下警告信息&#xff1a; 首先下载…

Java EE并发API教程

这是一个示例章节&#xff0c;摘自Francesco Marchioni编辑的WildFly上的实用Java EE 7开发 。 本章讨论了新的Java EE并发API&#xff08;JSR 236&#xff09; &#xff0c;它概述了使用一组托管资源在Java EE容器上并行执行任务的标准方法。 为了描述如何在您的应用程序中使…

经桥科技与湖南文化艺术产业集团合力打造“网乐潇湘”

签约仪式现场 签约仪式现场 签约仪式现场 经网1月7日讯(记者 陈飞 曹亮) 湖南经桥网络科技有限公司与湖南文化艺术产业集团“网乐潇湘”项目签约仪式隆重举行。经桥科技董事长姜志强、湖南文化艺术产业集团总经理陈介辉等领导出席此次签约仪式。 经桥科技与湖南文化艺术产业集…

带有Angular JS的Java EE 7 –第1部分

今天的帖子将向您展示如何使用Java EE 7和Angular JS构建非常简单的应用程序。 在去那里之前&#xff0c;让我告诉您一个简短的故事&#xff1a; 我不得不承认&#xff0c;我从来都不是Java语言的忠实拥护者&#xff0c;但是我仍然记得我第一次使用它。 我不记得确切的年份&am…

深入理解CSS定位中的堆叠z-index

前面的话 对于所有定位&#xff0c;最后都不免遇到两个元素试图放在同一位置上的情况。显然&#xff0c;其中一个必须盖住另一个。但&#xff0c;如何控制哪个元素放在上层&#xff0c;这就引入了属性z-index 定义 利用z-index&#xff0c;可以改变元素相互覆盖的顺序。这个属性…