关于内存不可见性的验证网上的例子很多,比如https://blog.csdn.net/weixin_33656548/article/details/88846777;
上文JDK版本jdk1.8.0_201,本人是jdk1.8.0_202;
public class VolatileTest {
public boolean isShutdown;
public boolean getShutdown () {
return isShutdown;
}
public void shutdown () {
isShutdown = true;
}
public class ReaderThread extends Thread {
@Override
public void run() {
try {
System.out.println("开始循环");
while (!isShutdown) {
};
System.out.println("结束循环");
} catch (Exception e) {
e.printStackTrace();
}
}
}
public class WatchThread extends Thread {
@Override
public void run() {
shutdown();
}
}
public static void main(String[] args){
try {
VolatileTest volatileTest = new VolatileTest();
volatileTest.new ReaderThread().start();
//让主线程睡眠一秒,确保另一个线程调用shutdown方法时死循环已经开始
Thread.sleep(1000);
volatileTest.new WatchThread().start();
//此刻的睡眠是为了确保shutdown方法对isShutdown变量的修改已经同步到主内存
Thread.sleep(1000);
//打印isShutdown的值
System.out.println("getShutdown:" + volatileTest.getShutdown());
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
运行结果,并未出现内存不可见现象:
因为手头没有下载jdk1.8.0_201版本,正好有jdk1.7(7u51);运行结果如下,出现内存不可见现象:
java9 运行结果:
jdk1.8.0_202 为啥与众不同呢?
Volatile缓存可见性实现原理:
底层实现原来主要是通过汇编lock前缀指令,它会锁定这块内存区域的缓存(缓存行锁定),写回到主内存
IA-32架构软件开发者手册对lock指令的解释:
1)会将当前处理器缓存的数据立即写回到系统内存;
2)这个写回内存的操作会引起其他CPU款村了该内存地址的数据无效(MESI协议)
首先开启MESI缓存一致性协议,实际上开启了总线嗅探机制,当线程二将数据修改isShutdown = true通过总线写入主内存时,其他线程CPU可以监听到数据的改变,从而将工作内存的数据去掉,重新从主内存去读取,也就能读取到最新的值isShutdown = true。
https://cloud.tencent.com/developer/article/1121728