1.轻量级锁:针对多个线程在不同时间访问同一把锁的情况
(1) 加锁:
- 线程在执行同步块之前,JVM会在当前线程的栈桢中创建用于存储锁记录的空间,并将对象头中的Mark Word复制到锁记录中,复制的这个称作Displaced Mark Word。
- 线程尝试使用CAS操作将对象头中的Mark Word替换为指向锁记录的指针。
- 如果成功,则成功获取锁;如果失败,表示有其他线程竞争锁,当前线程则自旋来获取锁。
(2) 解锁:
- 轻量级锁解锁时,会使用CAS操作将Displaced Mark Word替换回对象头;
- 若替换成功,说明没有竞争,成功释放;
- 若替换失败,说明此时存在竞争,则升级为重量级锁。
(3) 轻量级锁状态下64位Mark Word:
2.重量级锁:针对多个线程同一时刻访问同一资源的情况
(1) 过程:
重量级锁其实就是JDK1.6以前的synchronized同步机制,基于monitor计数器机制,JVM会阻塞加锁失败的线程,当锁释放之后,再唤醒这些阻塞的线程。JVM采用自适应自旋来减少性能损失。
(2) 重量级锁状态下64位Mark Word:
3.锁粗化:
锁粗化其实就是将多次拼接在一起的加锁、解锁操作合并为一次,将多个连续的锁扩展为一个范围更大的锁。代码示例如下:
public class StrTest {
public static void main(String[] args) {
StringBuffer str = new StringBuffer();
str.append(1);
str.append(2);
str.append(3);
System.out.println(str);
}
}
上述代码其实在每次append()时,都会进行加锁、解锁操作,当虚拟机检测到有一连串的对同一个对象的加锁解锁操作时,就会在第一次append()时加锁,最后一次append()后解锁。如下代码:
public class StrTest<