JMM是Java为解决多线程可见性、有序性、原子性问题制定的内存模型规则,规定变量存于主内存,线程操作需通过工作内存读写,volatile仅保证可见性与禁止重排序,不保证原子性与互斥,happens-before是判断操作可见性的逻辑先行关系。
JMM不是物理内存结构,而是Java为解决多线程下“变量改了别人看不见、执行顺序乱了、操作一半被插队”这三类问题,定的一套规则。 它不描述堆栈在哪,只规定:变量必须存在主内存;每个线程操作前得先拷一份到自己的工作内存;改完必须写回去,读之前得重新拉最新值——否则就可能卡死、算错、永远等不到结果。
ready = true 后另一个线程还在死循环?这是最典型的可见性失效场景。线程A把 ready 写成 true,但只更新了自己工作内存,没刷回主内存;线程B一直从自己工作内存里读 ready,始终是 false。
volatile 和 final 字段也在此)read → load → use → assign → store → write,中间任何一步缺失或延迟,都可能导致不一致volatile 怎么破局?它管什么、不管什么?volatile 是JMM提供的轻量级同步机制,它直接作用于这8个原子操作中的 read/load 和 assign/stor/
write,强制每次读都从主内存拉、每次写都立刻刷主内存。
volatile 变量,其他线程后续读一定能见到新值(靠 volatile 变量规则的 happens-before 保障)volatile 变量的读写“挪”到临界区外(如单例双重检查锁中防止对象未构造完成就被引用)count++ 即使 count 是 volatile,仍是“读-改-写”三步,仍可能丢更新;要用 AtomicInteger 或 synchronized
volatile 标记的临界区,不会排队,只是“大家都能看到最新状态”它不是内存操作,而是一组**逻辑先行关系**,用来判断“操作A的结果是否对操作B可见”。只要满足任一 happens-before 规则,JMM就保证A对B可见且不重排序——开发者不用纠结底层怎么刷缓存,只看代码逻辑是否符合这些规则。
happens-before 后面的语句(即使被重排序,效果也等价)unlock() happens-before 后续任意线程的 lock()
volatile 变量规则:volatile 写 happens-before 后续任意线程对该变量的读Thread.start() happens-before 新线程的任意动作happens-before B,B happens-before C,则 A happens-before C真正容易被忽略的是:happens-before 是**充分不必要条件**——不满足它不一定出错(比如碰巧缓存同步了),但一旦出错,几乎肯定是因为它断了。调试并发Bug时,第一反应不该是加日志,而是画出关键变量的 happens-before 链,看哪一环断了。