public class Test implements Runnable{
private static long field = 0;
private volatile long value;
public long getValue() {
return value;
}
public void setValue(long value) {
this.value = value;
}
public Test(long value) {
this.setValue(value);
}
@Override
public void run() {
int i = 0;
while(true){
while (i < 10000000) {
Test.field = this.getValue();
long temp = Test.field;
if (temp != 1L && temp != -1L) {
System.out.println("出现错误结果" + temp);
System.exit(0);
}
i++;
}
System.out.println("运行正确");
}}
public static void main(String[] args) throws InterruptedException {
// 获取并打印当前JVM是32位还是64位的
String arch = System.getProperty("sun.arch.data.model");
System.out.println(arch+"-bit");
Test t1 = new Test(1);
Test t2 = new Test(-1);
Thread T1 = new Thread(t1);
Thread T2 = new Thread(t2);
T1.start();
T2.start();
T1.join();
T2.join();
}}
解释:
如果jvm是64位的,那么field只能是1或-1.
如果jvm是32位的,那么
对于long和double变量,把他们作为2个原子性的32位值来对待,而不是一个原子性的64位值,
这样将一个long型的值保存到内存的时候,可能是2次32位的写操作,
2个竞争线程想写不同的值到内存的时候,可能导致内存中的值是不正确的结果。
1、写入高位32位值(线程2)
2、写入高位32位值(线程1)
3、写入低位32位值(线程1)
4、写入低位32位值(线程2)
即结果可能出现
32-bit出现错误结果-4294967295运行正确
即有线程t2先进行了写入高32位值,然后被中断,t1又进行写入高位和低位32位值,被中断,然后t1又写入低32位值,导致t2中的field值前32位是0,后32位是1所以结果是4294967295为什么是负呢,暂时不清楚