1、轻量级锁
使用场景:
如果一个对象虽然有多线程访问,但多线程访问的时间是错开的(也就是没有竞争),那么可以使用轻量级锁来优化
轻量级锁对使用者是透明的,即语法仍然是synchronized
假设有两个方法同步块,利用同一个对象加锁
static final Object obj = new Object();
public static void method1() {
synchronized(obj) {
//同步块A
method2();
}
}
public static void method2() {
synchronized(obj) {
//同步块B
}
}
- 创建锁记录(Lock Record)对象,每个线程的栈帧都会包含一 个锁记录的结构,内部可以储存锁定对象的Mark Word
- 让锁记录中Object reference 指向锁对象,并尝试用cas替换Object 的Mark Word,将Mark Word 的值存入锁记录
- 如果cas替换成功,对象头中存储了锁记录地址和状态00,表示由该线程给对象加锁,这时图示如下
-
如果cas失败,有两种情况
- 如果是其它线程已经持有了该Object的轻量级锁,这时表明有竞争,进入锁膨胀过程
- 如果是自己执行了synchronized锁重入,那么再添加一条Lock Record作为重入的计数
- 当退出synchronized代码块(解锁时)如果有取值为null代锁记录,表示有重入,这时重置记录,表示
重入计数减一
- 当退出synchronized代码块(解锁时)锁记录的值不为null,这时使用cas将Mark Word的值恢复给对象头
- 成功,则解锁成功
- 失败,说明轻量级锁进行了锁澎湃或已经升级为重量级,进入重量级锁解锁流程