多线程—轻量级锁

使用场景:如果一个对象被多个线程使用,但是每个线程使用对象的时间是错开的,也就是多个线程之间没有竞争,那么,可以使用轻量级锁来优化。

语法:使用synchronized

场景:假设两个方法同步块,对同一个对象加锁。

private static final Object object = new Object();
public static void methodOne(){
    synchronized(onject){
        //同步代码块
        methodTwo;
    }
}
public static void methodTwo(){
    synchronized(onject){
        //同步代码块
    }
}
  • 线程的每个方法运行时,都会产生一个栈帧,每个栈帧都包含一个锁记录的结构(Lock Reord),内部可以存储锁定对象的Mark Work值。

  •  让所记录中的Object reference指向锁对象,并尝试用cas替换(原子性操作)Object中的Mark work,将Mark word中的值存入锁记录(lock record)

  • 如果cas成功,那么对象头里面就存储的是锁记录地址和状态00,表示由该线程给对象加锁;解锁就是将所记录地址和状态复原。

 

 如果对象头里面的状态已经变为了00,那么意味着已经有线程持有了改对象的轻量级锁,如果这是有其他线程获取对象锁,这时存在竞争关系,进入锁膨胀过程。

  • 开头场景的这种情况:一个线程的两个方法,都锁同一个对象,并且其中一个方法已经获取了轻量级锁,那么另外一个方法再次获取锁的时候,就被称为锁重入

锁重入时,Lock Record 中的lock地址值就不和object中的Mark Word进行交换了。 

锁重入的解锁步骤:

1、判断Lock Record 里面的lock地址和状态码是否为空,如果为空,就直接锁数量减一。

2、如果不为空,使用cas将lock Record地址和状态码恢复。

        a.cas成功,则解锁成功。

        b.cas失败,说明轻量级锁进入了锁膨胀(将轻量级锁升级为重量级锁),进入重量级的解锁过程。

Guess you like

Origin blog.csdn.net/qq_42251944/article/details/120862165