自旋锁基本原理
轻量级锁在加锁过程中,用到了自旋锁
所谓自旋,就是指当有另外一个线程来竞争锁时,这个线
程会在原地循环等待,而不是把该线程给阻塞,直到那个
获得锁的线程释放锁之后,这个线程就可以马上获得锁的。
注意,锁在原地循环的时候,是会消耗 cpu 的,就相当于
在执行一个啥也没有的 for 循环。
所以,轻量级锁适用于那些同步代码块执行的很快的场景,
这样,线程原地等待很短的时间就能够获得锁了。
自旋锁的使用,其实也是有一定的概率背景,在大部分同
步代码块执行的时间都是很短的。所以通过看似无异议的
循环反而能提升锁的性能。
但是自旋必须要有一定的条件控制,否则如果一个线程执行同步代码块的时间很长,那么这个线程不断的循环反而
会消耗 CPU 资源。默认情况下自旋的次数是 10 次,
可以通过 preBlockSpin 来修改在 JDK1.6 之后,引入了自适应自旋锁,自适应意味着自旋
的次数不是固定不变的,而是根据前一次在同一个锁上自
旋的时间以及锁的拥有者的状态来决定。
如果在同一个锁对象上,自旋等待刚刚成功获得过锁,并
且持有锁的线程正在运行中,那么虚拟机就会认为这次自
旋也是很有可能再次成功,进而它将允许自旋等待持续相
对更长的时间。如果对于某个锁,自旋很少成功获得过,
那在以后尝试获取这个锁时将可能省略掉自旋过程,直接
阻塞线程,避免浪费处理器资源
轻量级锁的升级
轻量级锁的锁释放逻辑其实就是获得锁的逆向逻辑,通过CAS 操作把线程栈帧中的 LockRecord 替换回到锁对象的MarkWord 中,如果成功表示没有竞争。如果失败,表示当前锁存在竞争,那么轻量级锁就会膨胀成为重量级锁