文章目录
-
- [1. 锁的本质:争夺共享资源](#1. 锁的本质:争夺共享资源)
- [2. 栈的封闭性(线程私有)](#2. 栈的封闭性(线程私有))
- [3. JIT 的"上帝视角"](#3. JIT 的“上帝视角”)
-
- 优化前(你的原始代码):
- [JIT 介入分析:](#JIT 介入分析:)
- 优化后(实际执行的代码):
- [4. 为什么要删掉?锁真的很贵吗?](#4. 为什么要删掉?锁真的很贵吗?)
- 总结
这是一个非常棒的逻辑推导点。要理解"为什么不逃逸就不需要锁",核心在于理解**"锁到底是给谁看的"**。
在并发编程中,锁(Synchronization)的唯一目的就是处理竞争(Contention) 。而竞争的前提是:多个线程能够同时看到并访问同一个对象。
1. 锁的本质:争夺共享资源
想象一下,如果世界上只有你一个人,你进自己家门还需要反锁吗?你上厕所还需要锁门吗?
显然不需要。因为锁是为了防止"别人"进入。
在 Java 中:
- 逃逸的对象 :像是一个公共厕所。任何线程(路人)都可以进去。为了保证同一时间只有一个人使用,必须在门口挂把锁。
- 不逃逸的对象 :就像是你大脑里的一个念头。除了你自己,没有任何人能感知到它,更别提抢夺它了。
2. 栈的封闭性(线程私有)
理解这个问题的关键在于 Java 的内存模型:
- 堆(Heap):是公共场所,所有线程共享。
- 栈(Stack) :是线程私有的。每个线程都有自己独立的栈。
如果逃逸分析确定一个对象不逃逸 ,那么 JVM 可能会发起"栈上分配"。这个对象被创建在当前线程的栈帧里。
由于其他线程根本无法访问到你的栈,这个对象就变成了当前线程的"绝对私产"。
结论:既然这个对象永远只会被当前线程访问,那么在这个对象上加锁、释放锁的操作就是 100% 浪费时间的废动作。
3. JIT 的"上帝视角"
JIT(即时编译器)在代码运行过程中,会像一个俯瞰全局的上帝,观察代码的执行模式。
优化前(你的原始代码):
java
public void syncMethod() {
// obj 是局部变量,不逃逸
Object obj = new Object();
synchronized(obj) {
// 执行逻辑
System.out.println("Hello");
}
}
JIT 介入分析:
- 分析 :
obj是在方法内部new出来的,且没有返回给调用者,没有赋值给静态变量,没有传给其他线程。 - 判断 :
obj确定不逃逸。 - 推导 :不逃逸 → \rightarrow → 只有当前线程能看到
obj→ \rightarrow → 绝对不存在竞争 → \rightarrow → 锁没意义。 - 动作 :执行锁消除。
优化后(实际执行的代码):
java
public void syncMethod() {
Object obj = new Object();
// 锁被直接拆掉了,没有加锁解锁的开销
System.out.println("Hello");
}
4. 为什么要删掉?锁真的很贵吗?
你可能会想:"就算没意义,留着锁也没事吧?"
非常有事。 锁的操作哪怕在没有竞争的情况下,也是有开销的:
- 内存屏障(Memory Barrier):为了保证可见性,锁操作会强制 CPU 刷新缓存,这会打断 CPU 的指令流水线。
- CAS 操作:轻量级锁至少需要一次 CAS 指令来更新对象头,CAS 比普通指令慢得多。
- 对象头修改:锁会修改对象头的 Mark Word。
通过逃逸分析删掉这些无用的锁,可以让这段代码的运行速度瞬间提升,达到和普通无锁代码完全一样的性能。
总结
"不逃逸"意味着"线程私有" 。
既然是私有的,就代表没有竞争对手。既然没有竞争对手,那锁就是一种纯粹的浪费。JIT 删掉它是为了去掉多余的指令,让 CPU 跑得更顺畅。
这其实体现了 Java 优化的一种核心思想:尽可能把"复杂的多线程环境"简化为"简单的单线程逻辑"来执行。