锁的状态 为了减少获得锁和释放锁带来的性能消耗,引入了“偏向锁”和“轻量级锁”,锁一共有4种状态,级别分为从低到高依次是,无锁状态,偏向锁状态,轻量级锁状态和重量级锁状态。锁可以升级但不能升级。
为了减少获得锁和释放锁带来的性能消耗,引入了“偏向锁”和“轻量级锁”,锁一共有4种状态,级别分为从低到高依次是,无锁状态,偏向锁状态,轻量级锁状态和重量级锁状态。锁可以升级但不能升级。
偏向锁
当一个线程访问同步块并获取锁时,会在对象头和栈帧中的锁记录里存储锁偏向的线程ID,==以后该线程在进入和退出同步块时不需要进行CAS操作来加锁和解锁,只需简单地测试一下对象头中的Mark word是否存储着指向当前线程的偏向锁。==如果测试成功,表示线程已经获得了锁。如果测试失败,则需要在测试一下Mark Word 中偏向锁的标识是否设置成1;如果没有设置,则使用CAS竞争锁;如果设置了,则尝试使用CAS将对象头的偏向锁指向当前线程。偏向锁的撤销流程如下图所示。
偏向锁的撤销
偏向锁使用了一种等到竞争出现才释放锁的机制,当其他线程竞争偏向锁时,持有偏向锁的线程才会释放。偏向锁的撤销需要等待全局安全点。首先会暂停持有偏向锁的线程, 然后检查持有偏向锁的线程是否存活, 如果线程不处于活动状态, 则将锁对象的对象头设置为无锁状态; 如果线程仍然活着, 则锁对象的对象头中的MarkWord和栈中的锁记录要么重新偏向于其它线程要么恢复到无锁状态, 最后唤醒暂停的线程(释放偏向锁的线程).
关闭偏向锁
如果应用程序所有的锁通常情况下处于竞争状态,可以通过JVM参数关闭偏向锁,那么默认会进入轻量级锁状态。
轻量级锁
轻量级锁加锁
线程在执行同步块之前,jvm先在当前线程的栈帧中创建用于存,记录的空间,并将对象头中的Mark Word复制到锁记录中,官方称为Displaced Mark Word。然后线程尝试使用CAS将对象头中的Mark Word替换为指向锁记录的指针。如果成功,当前线程获得锁,如果失败,表示其他线程竞争锁,当前线程便尝试使用自旋来获取锁。
轻量级锁解锁
轻量级解锁时,会使用原子的CAS操作将Displaced Mark Word替换到对象头。如果成功,则表示没有竞争发生。如果失败,表示当前锁存在竞争,锁就会膨胀成重量级锁。