您的位置:首页 >Java内存模型详解:并发底层原理剖析
发布于2026-03-16 阅读(0)
扫一扫,手机访问
Java内存模型(JMM)是定义多线程下变量可见性、有序性及重排序规则的抽象规范,非物理内存布局;其核心为“主内存-工作内存”模型,通过volatile、synchronized和final提供不同强度的内存语义保障。

很多人一看到“内存模型”就下意识联想到堆、栈、方法区这些运行时数据区——这是常见误解。JMM 不描述 JVM 实际怎么分配内存,而是定义了多线程环境下,变量读写操作 在不同线程间如何可见、何时有序、哪些重排序被允许。它的核心是抽象的“主内存-工作内存”模型,每个线程有自己的工作内存(可理解为寄存器或CPU缓存的抽象),所有共享变量都存在于主内存中。
关键点在于:线程对变量的所有操作(读、写、加锁、解锁)都必须通过工作内存与主内存交互,不能直接读写主内存。这就引出了可见性、原子性、有序性三大问题。
volatile 是 JMM 最轻量级的同步机制,但它只保证两点:变量对所有线程的可见性,以及禁止特定类型的指令重排序。它不保证复合操作的原子性(比如 i++)。
编译器和处理器对 volatile 变量的读写有明确约束:
StoreStore + StoreLoad 屏障)LoadLoad + LoadStore 屏障)例如:volatile boolean flag = false; 后续的 int x = 10; 不会被重排到 flag = true 之前——这正是 DCL(双重检查锁定)中 instance = new Singleton() 需要 volatile 的根本原因。
synchronized 的语义远不止“加锁”。在 JMM 中,它通过 happens-before 规则建立线程间的偏序关系:一个线程对锁的 unlock 操作 happens-before 另一个线程对该锁的 lock 操作。
这意味着:
注意:synchronized 块的锁对象必须是同一个实例才有意义;锁的粒度影响性能,但不影响 JMM 语义本身。另外,wait()/notify() 也参与 happens-before 链——notify() happens-before 被唤醒线程的 wait() 返回。
final 字段不只是“不可变”,它在构造完成那一刻就向其他线程提供了安全发布保障。JMM 规定:如果一个对象被正确构造(即构造过程中 this 引用未逸出),那么其他线程看到该对象的 final 字段时,一定能看到构造器中为其设置的值,且不会看到默认值(如 0、null)。
这个保障依赖于两个动作的禁止重排序:
换句话说,只要没发生 this 逃逸,final int x = 42; 的写入对其他线程就是立即可见的——不需要 volatile 或 synchronized。这是实现无锁不可变对象(如 String、LocalDateTime)的底层基础。
售后无忧
立即购买>office旗舰店
售后无忧
立即购买>office旗舰店
售后无忧
立即购买>office旗舰店
售后无忧
立即购买>office旗舰店
正版软件
正版软件
正版软件
正版软件
正版软件
1
2
3
7
9