第十八篇:第五篇中volatile意外问题的正确分析解答(含代码)

本文探讨了Java中volatile变量的可见性问题,特别是在多线程环境下,即使未标记为volatile的变量也可能展现出最新的值的现象。通过代码示例和深入分析,揭示了JVM内存模型对volatile变量的处理机制。

一文中遗留了一个问题,就是volatile只修饰了missedIt变量,而没修饰value变量,但是在线程读取value的值的时候,也读到的是最新的数据。但是在网上查了很多资料都无果,看来很多人对volatile的规则并不是太清晰,或者说只停留在很表面的层次,一知半解。

    这两天看《深入Java虚拟机——JVM高级特性与最佳实践》第12章:Java内存模型与线程,并在网上查阅了Java内存模型相关资料,学到了不少东西,尤其在看这篇文章的volatile部分的讲解之后,算是确定了问题出现的原因。

    首先明确一点:假如有两个线程分别读写volatile变量时,线程A写入了某volatile变量,线程B在读取该volatile变量时,便能看到线程A对该volatile变量的写入操作,关键在这里,它不仅会看到对该volatile变量的写入操作,A线程在写volatile变量之前所有可见的共享变量,在B线程读同一个volatile变量后,都将立即变得对B线程可见。

   回过头来看文章中出现的问题,由于程序中volatile变量missedIt的写入操作在value变量写入操作之后,而且根据volatile规则,又不能重排序,因此,在线程B读取由线程A改变后的missedIt之后,它之前的value变量在线程A的改变也对线程B变得可见了。

     我们颠倒一下value=50和missedIt=true这两行代码试下,即missedIt=true在前,value=50在后,这样便会得到我们想要的结果:value值的改变不会被看到。

    这应该是JDK1.2之后对volatile规则做了一些修订的结果。


    修改后的代码如下:

[java]  view plain  copy
  1. public class Volatile extends Object implements Runnable {  
  2.     //value变量没有被标记为volatile  
  3.     private int value;    
  4.     //missedIt变量被标记为volatile  
  5.     private volatile boolean missedIt;  
  6.     //creationTime不需要声明为volatile,因为代码执行中它没有发生变化  
  7.     private long creationTime;   
  8.   
  9.     public Volatile() {  
  10.         value = 10;  
  11.         missedIt = false;  
  12.         //获取当前时间,亦即调用Volatile构造函数时的时间  
  13.         creationTime = System.currentTimeMillis();  
  14.     }  
  15.   
  16.     public void run() {  
  17.         print("entering run()");  
  18.   
  19.         //循环检查value的值是否不同  
  20.         while ( value < 20 ) {  
  21.             //如果missedIt的值被修改为true,则通过break退出循环  
  22.             if  ( missedIt ) {  
  23.                 //进入同步代码块前,将value的值赋给currValue  
  24.                 int currValue = value;  
  25.                 //在一个任意对象上执行同步语句,目的是为了让该线程在进入和离开同步代码块时,  
  26.                 //将该线程中的所有变量的私有拷贝与共享内存中的原始值进行比较,  
  27.                 //从而发现没有用volatile标记的变量所发生的变化  
  28.                 Object lock = new Object();  
  29.                 synchronized ( lock ) {  
  30.                     //不做任何事  
  31.                 }  
  32.                 //离开同步代码块后,将此时value的值赋给valueAfterSync  
  33.                 int valueAfterSync = value;  
  34.                 print("in run() - see value=" + currValue +", but rumor has it that it changed!");  
  35.                 print("in run() - valueAfterSync=" + valueAfterSync);  
  36.                 break;   
  37.             }  
  38.         }  
  39.         print("leaving run()");  
  40.     }  
  41.   
  42.     public void workMethod() throws InterruptedException {  
  43.         print("entering workMethod()");  
  44.         print("in workMethod() - about to sleep for 2 seconds");  
  45.         Thread.sleep(2000);  
  46.         //仅在此改变value的值  
  47.         missedIt = true;  
  48. //      value = 50;  
  49.         print("in workMethod() - just set value=" + value);  
  50.         print("in workMethod() - about to sleep for 5 seconds");  
  51.         Thread.sleep(5000);  
  52.         //仅在此改变missedIt的值  
  53. //      missedIt = true;  
  54.         value = 50;  
  55.         print("in workMethod() - just set missedIt=" + missedIt);  
  56.         print("in workMethod() - about to sleep for 3 seconds");  
  57.         Thread.sleep(3000);  
  58.         print("leaving workMethod()");  
  59.     }  
  60.   
  61. /* 
  62. *该方法的功能是在要打印的msg信息前打印出程序执行到此所化去的时间,以及打印msg的代码所在的线程 
  63. */  
  64.     private void print(String msg) {  
  65.         //使用java.text包的功能,可以简化这个方法,但是这里没有利用这一点  
  66.         long interval = System.currentTimeMillis() - creationTime;  
  67.         String tmpStr = "    " + ( interval / 1000.0 ) + "000";       
  68.         int pos = tmpStr.indexOf(".");  
  69.         String secStr = tmpStr.substring(pos - 2, pos + 4);  
  70.         String nameStr = "        " + Thread.currentThread().getName();  
  71.         nameStr = nameStr.substring(nameStr.length() - 8, nameStr.length());      
  72.         System.out.println(secStr + " " + nameStr + ": " + msg);  
  73.     }  
  74.   
  75.     public static void main(String[] args) {  
  76.         try {  
  77.             //通过该构造函数可以获取实时时钟的当前时间  
  78.             Volatile vol = new Volatile();  
  79.   
  80.             //稍停100ms,以让实时时钟稍稍超前获取时间,使print()中创建的消息打印的时间值大于0  
  81.             Thread.sleep(100);    
  82.   
  83.             Thread t = new Thread(vol);  
  84.             t.start();  
  85.   
  86.             //休眠100ms,让刚刚启动的线程有时间运行  
  87.             Thread.sleep(100);    
  88.             //workMethod方法在main线程中运行  
  89.             vol.workMethod();  
  90.         } catch ( InterruptedException x ) {  
  91.             System.err.println("one of the sleeps was interrupted");  
  92.         }  
  93.     }  
  94. }  
    执行结果如下:


   很明显,这其实并不符合使用volatile的第二个条件:该变量要没有包含在具有其他变量的不变式中。因此,在这里使用volatile是不安全的。




附上一篇讲述volatile关键字正确使用的很好的文章:http://www.ibm.com/developerworks/cn/java/j-jtp06197.html

``` {---------- 系统参数定义 ----------} PE_CALC := IF(FINANCE(33)>0, FINANCE(30)/FINANCE(4), 1000); PB_CALC := IF(FINANCE(5)>0, FINANCE(30)/FINANCE(5), 1000); {---------- 动态参数优化模块(增强版)----------} VOLAT := STD(CLOSE,20)/MA(CLOSE,20); VAR_PD := IF(VOLAT<0.06,89,IF(VOLAT<0.12,55,IF(VOLAT<0.2,34,21))); MACD_F := MAX(5, CEILING(VAR_PD*0.382)); MACD_S := MIN(60, FLOOR(VAR_PD*1.618)); MACD_M := 9; {---------- 核心指标系统(优化计算)----------} FAST_EMA := EMA(CLOSE,MACD_F); SLOW_EMA := EMA(CLOSE,MACD_S); DIF := EMA(FAST_EMA,3) - EMA(SLOW_EMA,3); DEA := EMA(DIF,MACD_M); MACD := 2*(DIF-DEA); MA5 := MA(CLOSE,5); MA10 := MA(CLOSE,10); MA20 := EMA(CLOSE,20); MA60 := EMA(CLOSE,60); {---------- 信号增强模块(增加量价验证)----------} MOM := EMA(CLOSE,5)/REF(EMA(CLOSE,21),5)-1; WK_CHG := (CLOSE-REF(CLOSE,5))/REF(CLOSE,5)*100; MOM_CFM := EMA(CLOSE,5)>EMA(EMA(CLOSE,5),13) AND CLOSE>HHV(CLOSE*0.7,40); VOL5 := MA(VOL,5); VOL20 := EMA(VOL,20); VOL_RTO := VOL/REF(VOL,1); FLOW_CD := VOL>VOL20*1.8 AND SUM(IF(CLOSE>REF(CLOSE,1),VOL*C,-VOL*C),5)/CAPITAL*100>3; {---------- 多维过滤条件(增加波动率验证)----------} TREND_CD := MA5>MA10 AND MA10>MA20 AND CLOSE>MA60; BREAK_CD := CROSS(DIF,DEA) AND MACD>REF(MACD,1) AND DIF>0; VALUE_CD := PE_CALC<30 AND PB_CALC<4.5; BOLL_W := (MA(CLOSE,20)+2*STD(CLOSE,20)-(MA(CLOSE,20)-2*STD(CLOSE,20)))/MA(CLOSE,20)*100; VOLAT_CD := BOLL_W>10 AND BOLL_W<18; CHIP_CD := FINANCE(7)/100 < 1.2; {---------- 多周期验证模块(修正后)----------} // 修改点1: 添加有效指数代码 'SH000001' WEEK_CLOSE := CALLSTOCK('SH000001', "VT.CLOSE", 6, -2); WEEK_SG := WEEK_CLOSE > REF(WEEK_CLOSE,1) AND MA(WEEK_CLOSE,5) > MA(WEEK_CLOSE,10) AND WEEK_CLOSE > EMA(WEEK_CLOSE,20); // 修改点2: 标准化参数设置 'SH000001' M30_CLOSE := CALLSTOCK('SH000001', "VT.CLOSE", 1, -2); M30_BRK := M30_CLOSE > EMA(M30_CLOSE,20) AND VOL > MA(VOL,5)*1.2 AND CROSS(M30_CLOSE, EMA(M30_CLOSE,50)); {---------- 预警信号生成(增加量价确认)----------} DAY_ALT := CROSS(MA(CLOSE,5),MA(CLOSE,10)) AND VOL>REF(VOL,1)*1.5 AND CLOSE>OPEN*1.02 AND TIME<150000; ROC21 := (CLOSE-REF(CLOSE,21))/REF(CLOSE,21)*100; RANK_RC := (ROC21-LLV(ROC21,120))/(HHV(ROC21,120)-LLV(ROC21,120)+0.0001)*100; {---------- BETA系数计算(优化算法)----------} INDEX_MA := STKINDI('SH000001','MA.MA1',0,6,0); BETA_A := SLOPE(CLOSE/INDEX_MA,60)*STD(CLOSE,60)/STD(INDEX_MA,60); BETA_CD := (1+(BETA_A-1)*0.3)<1.2 AND CORR(CLOSE,INDEX_MA,60)>0.8; {---------- 信号综合判断(优化触发逻辑)----------} COND_CNT := VALUE_CD + VOLAT_CD + CHIP_CD; FINAL_SG := FILTER( TREND_CD AND BREAK_CD AND FLOW_CD AND COND_CNT>=2 AND MOM_CFM AND RANK_RC>80 AND DAY_ALT AND BETA_CD AND WEEK_SG AND M30_BRK, 3); {---------- 输出模块(增加信号强度标识)----------} ALERT(FINAL_SG,'多周期共振'), SOUND; DRAWTEXT(FINAL_SG, LOW,'★'), COLORYELLOW, LINETHICK3; FILTER(FINAL_SG,1), NODRAW;```你的身份是高级编程技术专家,精通各类编程语言,能对编程过程中的各类问题进行分析解答。我的问题是【我正在编辑【通达信量化择时选股】代码,遇到了 【错误句 : 详细信息 : 单词最大字符数不得超过 16 个 错误起始位置 : 1461 ; 长度: 7】,请帮我检查并改正错误点补全正确代码,原有选股逻辑完整保留,所有参数计算关系和信号触发条件优化计算,生成修正后完整代码
03-23
/**********************按键实验*********************************/ // 公司名称 :保定飞凌嵌入式技术有限公司 // 描 述 :按键控制蜂鸣器 // 版 权 :保定飞凌嵌入式技术有限公司 // 网 址 :www.witech.com.cn /***************************************************************/ /* 本实验接口说明 GPB5 ------ LED0 GPB6 ------ LED1 GPB8 ------ LED2 GPB10 ------ LED3 */ /*------------------------地址声明---------------------------*/ #define GPBCON (*(volatile unsigned *)0x56000010) #define GPBDAT (*(volatile unsigned *)0x56000014) #define GPBUP (*(volatile unsigned *)0x56000018) #define uchar unsigned char #define uint unsigned int /*-----------------------定义全局变量------------------------*/ /*-----------------------函数声明----------------------------*/ void Delay(int x); /*------------------------------------------------------------/ 函数名称: Delay 功能描述: 延时函数 传 参: int x 返 回 值: 无 -------------------------------------------------------------*/ void Delay(int x) { int k, j; while(x) { for (k=0;k<=0xff;k++) for(j=0;j<=0xff;j++); x--; } } /*------------------------------------------------------------- 函数名称: ledMain 功能描述: 入口程序 初始化后,进入跑马灯死循环 传 参: 无 返 回 值: int 0 -------------------------------------------------------------*/ int ledMain(void) { GPBCON = 0x1dd7fc; // GPB5,GPB6,GPB8,GPB10设置为输出 GPBDAT = ((1<<5)|(1<<6)|(1<<8)|(1<<10)); //使LED全灭 GPBDAT&=0xffe; //关闭蜂鸣器 GPBUP = 0x00; while (1) // 死循环 { GPBDAT = ~(1<<5); //LED0亮 Delay(500); GPBDAT = ~(1<<6); //LED1亮 Delay(500); GPBDAT = ~(1<<8); //LED2亮 Delay(500); GPBDAT = ~(1<<10); //LED3亮 Delay(500); } return 0; } ; init.s AREA |DATA|,CODE,READONLY ENTRY ldr r13, =0x1000 ;设置堆栈栈顶指针 IMPORT ledMain b ledMain END 1. 预定义语句#define rGPBCON (*(volatile unsigned *)0x56000010)中,关键字volatile的作用是什么?unsigned * 的作用的是什么? 第一个*的作用是什么?rGPBCON代表的是什么特殊功能寄存器?其功能是什么? volatile 不稳定的,易变的,告诉编译器不要对volatile 修饰的变量或存储单元进行优化,其值随时可变,每次都直接访问存储单元。 unsigned * 的作用是强制类型转换,把0x56000010的本来的整数类型强制转换成无符号指针类型 第一个*的作用在地址0x56000010上,是指针运算符,访问地址为0x56000010的特殊功能寄存器 GPBCON是 GPIO的B端口的配置寄存器,为每一个GPIO B组的引脚定义功能(用2位定义一个引脚的功能) 00 - 输入 01-输出 10-定义具体硬件模块引脚功能 11 - 保留 2. 把十六进制数0x1dd7fc转换成二进制,配置管脚GPB5的功能是什么?是GPBCON的哪些位段以及什么数值决定的? 3. 语句rGPBDAT = ((1<<5)|(1<<6)|(1<<8)|(1<<10)); 执行后,GPB5、GPB6、GPB8、GPB10分别输出高电平还是低电平? 高电平, 对应的LED灯亮还是灭? 都处于熄灭状态(初始状态) 4. 语句rGPBDAT = ~(1<<8); 执行后,GPB5、GPB6、GPB8、GPB10分别输出高电平还是低电平?对应的LED灯亮还是灭? GPB5、GPB6、GPB10输出高电平,对应的LED灯熄灭;GPB8输出低电平,对应的LED灯点亮
最新发布
06-08
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值