锁升级
无锁
- 无竞争情况
- 存在竞争,那就用非锁方式实现同步线程
- 锁标志位为01
偏向锁
- 只有一个线程在使用变量
- Mark Word的前23bit为使用变量的线程ID
- 锁标志位为01
偏向锁的使用过程:
- 检查Mark Word里面是否存储着当前线程的ID
- 如果有,那么说明已经获得了锁,直接使用
- 如果没有,测试一下偏向锁的标记是否为1
- 如果没有就直接用CAS去竞争锁
- 如果有就用CAS去尝试把线程的ID换到Mark Word当中
- 如果成功就直接使用
- 如果不成功,撤销偏向锁,暂停线程(需要等到全局安全点,在这个时间点没有正在执行的字节码)
- 检查偏向锁的那个线程是否活着,如果没活着就设置为把对象头设置为无锁状态,如果还活着,拥有偏向锁的栈会被执行,遍历偏向对象的锁记录,栈中的所锁记录和对象头的Mark Word要么重新偏向其他线程,要么恢复到无锁或者标记对象不适合偏向锁,最后唤醒暂停的线程;
总结:偏向锁使用了一直,直到存在竞争才释放锁的机制
轻量级锁
轻量锁的加锁
- JVM会在当前线程的栈中开辟一段Lock Record的空间,其中有两部分,一部分存在Mark Word的副本,叫做DIsplaced Mark Word,还有一部分存放对象的指针owner。
- 然后线程尝试使用CAS把对象头中的对象Mark Word中的前30位会生成一个指针指向Lock Record 的指针;
- 如果成功了,当前线程获得锁,
- 如果失败了表示其他线程竞争锁,那么线程就会自旋去获得锁。
上述步骤之后,对象和锁就进行了绑定
如果出现有线程要访问,那么它就会自旋,也就是进行轮询;
后面优化变成了适应性自旋,如果之前获得过锁那么自旋时间会提高;
轻量锁的释放
- 使用原子的CAS操作将Displace Mark Word替换回对象中。
- 如果成功,说明没有竞争。
- 如果失败,表示存在竞争,锁就会膨胀成为重量级锁。
重量级锁
真正的锁,使用monitor,CPU消耗大。