代码如下
public static void main(String[] args) { Test a = new Test(); a.start(); for (; ; ) { if (a.isFlag()) { System.out.println("1"); } } } static class Test extends Thread { private boolean flag = false; public boolean isFlag() { return flag; } @SneakyThrows @Override public void run() { Thread.sleep(1000); flag = true; System.out.println(flag); } } 背景
野生码仔,对这个问题困惑了一下午
我所了解的知识(不一定正确)
- jmm 内存模型中有主存和线程工作内存之分,线程读取一个变量会从主存读到工作内存之中,然后一切操作都是基于工作内存
- 工作内存是逻辑概念,实际可能是比如 cpu 缓存
- cpu 缓存一致性协议,如果有写操作的话,会通知主存此变量失效,然后其他线程有用这个变量的话会重新读取
代码执行结果
主线程中读到的 flag 值始终为 false
补充
代码改为如下,加上了 else
for (; ; ) { if (a.isFlag()) { System.out.println("1"); }else{ System.out.println("2"); } } a 线程修改完 flag 值后,主线程是能拿到最新的值的
问题
- else 到底影响了主存和工作内存之间的哪些交互?
- 在没有 else 的情况下,a 线程修改了 flag 的值,main 线程的死循环里为何一直拿不到修改后的值
猜测
是否和 cpu 缓存使用的 mesi 协议有关?
