对象锁锁实例(this),保护实例变量;类锁锁Class对象,保护静态资源;二者互不干扰,选择依据是操作数据的归属:实例数据用对象锁,静态数据用类锁。

对象锁和类锁本质都是 synchronized 锁,区别在于“锁住谁”——对象锁锁的是某个具体实例(this),类锁锁的是整个类的 Class 对象(如 MyClass.class)。它们互不干扰,适用场景完全不同。
对象锁:锁住单个实例,保护该实例的数据
对象锁用于同步同一个对象实例上的多个方法或代码块。只要线程操作的是同一个对象,就会排队执行;如果是不同对象,就完全不冲突。
- 修饰非静态方法:
public synchronized void doWork() { ... }→ 锁的是调用该方法的对象(this) - 同步代码块显式锁 this:
synchronized(this) { ... }→ 效果等同于上一条 - 同步代码块锁其他实例变量:
synchronized(obj) { ... }→ 只要obj是同一引用,就构成对象锁
例如:两个线程分别调用 new Counter().increment() 和 new Counter().increment(),因对象不同,不会互斥;但若都调用同一个 counter.increment(),则会串行执行。
类锁:锁住整个类,保护静态资源或全局状态
类锁作用于类本身,所有线程访问该类的静态同步方法或 synchronized(XXX.class) 块时,必须竞争同一把锁。不管创建多少个对象实例,这把锁只有一份。
立即学习“Java免费学习笔记(深入)”;
- 修饰静态方法:
public static synchronized void initConfig() { ... }→ 锁的是MyClass.class - 同步代码块锁 Class 对象:
synchronized(MyClass.class) { ... }→ 显式指定类锁
典型用途包括:单例模式的双重检查锁、静态计数器、类级别的初始化逻辑。比如多个线程同时调用 ConfigLoader.load()(静态同步方法),只会有一个线程真正执行加载,其余等待。
对象锁和类锁可以共存,互不影响
一个类里既有对象锁方法,又有类锁方法,它们使用的是两把完全独立的锁:
- 线程 A 调用
obj.doInstanceJob()(对象锁)→ 拿到obj的锁 - 线程 B 同时调用
MyClass.doStaticJob()(类锁)→ 拿到MyClass.class的锁 - 两者可并行执行,因为锁对象不同
但要注意:如果在对象锁方法内部又去访问类锁资源(比如读写静态变量),仍需额外考虑是否需要同步,否则可能引发竞态。
怎么选?看你要保护的数据属于谁
判断依据很简单:
- 操作的是 实例变量(如
this.count)、或只影响当前对象的行为 → 用对象锁 - 操作的是 静态变量(如
static int totalCount)、或涉及全局限制/初始化 → 用类锁 - 不确定时,优先考虑最小作用域:能用对象锁就不轻易升级为类锁,避免过度串行化










