synchronized锁升级

2年前 (2022) 程序员胖胖胖虎阿
300 0 0

synchronized锁升级

目录

      • synchronized锁升级
        • 1.synchronized锁本质
          • 1.1 Java层面
          • 1.2 字节码层面
          • 1.3 JVM层面(对象头)
        • 2.synchronized锁升级
          • 2.1 无锁
          • 2.2 偏向锁
          • 2.3 轻量级锁
          • 2.4 重量级锁

1.synchronized锁本质

​ synchronized锁本质是一个对象锁,即在对象中锁的一个过程。

1.1 Java层面

​ 在Java层面上加锁,一般有三种方式:

  • synchronized同步代码块

    //同步代码块会锁住o对象
    synchronized(Object o){
        
    }
    
  • synchronized修饰普通方法:

    //加在普通方法前锁的是this对象,即调用method方法的对象
    public synchronized void method(){
        
    }
    
  • synchronized修饰静态方法:

    //加在静态方法前锁的是这个类对象
    public static synchronized void method(){
        
    }
    
1.2 字节码层面

​ 在java中加入synchronized锁之后,编译成字节码之后会是怎么样的呢?

  • synchronized修饰方法

    //sunchronized代码块形式
    //在字节码中会有一个修饰符:ACC_SYNCHRONIZED
    public synchronized void method(){
        
    }
    
  • synchronized代码块

    //这种加锁方式,在字节码层面是:monitorenter和monitorexit
    //进入同步方法时,会有monitorenter指令
    //出去同步方法时,会有monitorexit指令
    synchronized(Object o){
        
    }
    
1.3 JVM层面(对象头)

​ 在JVM中,每个对象都有一个对象头,synchronized用的锁是存在对象头中的,以下是对象头里Mark Word的存储结构

锁状态 25bit 4bit 1bit是否为偏向锁 2bit锁标志位
无锁状态 对象的hashcode 对象分代年龄 0 01

​ 在运行期间,Mark Word中的数据会随着锁标志位的变化而变化,Mark Word可能会变成以下四种状态:

偏向锁

锁状态 23bit 2bit 4bit 1bit(是否为偏向锁) 2bit(锁标志位)
偏向锁 线程ID Epoch 对象分代年龄 1 01
锁状态 23bit+2bit+4bit+1bit 2bit(锁标志位)
轻量级锁 指向栈中锁记录的指针 00
重量级锁 指向互斥量(重量级锁)的指针 10

2.synchronized锁升级

​ 根据上面内容可以知道,synchronized锁有四种状态:无锁,偏向锁、轻量级锁和重量级锁,下面介绍四种状态和其之间的转换。

2.1 无锁

​ 当一个对象被创建之后,还没有线程进入,这个时候对象处于无锁状态,其Mark Word中的信息如上表所示。

2.2 偏向锁

​ 当锁处于无锁状态时,有一个线程A访问同步块并获取锁时,会在对象头和栈帧中的锁记录记录线程ID,以后该线程在进入和退出同步块时不需要进行CAS操作来进行加锁和解锁,只需要简单的测试一下啊对象头中的线程ID和当前线程是否一致。

2.3 轻量级锁

​ 在偏向锁的基础上,又有另外一个线程B进来,这时判断对象头中存储的线程A的ID和线程B不一致,就会使用CAS竞争锁,并且升级为轻量级锁,会在线程栈中创建一个锁记录(lock Record),将Mark Word复制到锁记录中,然后线程尝试使用CAS将对象头的Mark Word替换成指向锁记录的指针,如果成功,则当前线程获得锁;失败,表示其他线程竞争锁,当前线程便尝试CAS来获取锁。

2.4 重量级锁

​ 当线程没有获得轻量级锁时,线程会CAS自旋来获取锁,当一个线程自旋10次之后,仍然未获得锁,那么就会升级成为重量级锁。

​ 成为重量级锁之后,线程会进入阻塞队列(EntryList),线程不再自旋获取锁,而是由CPU进行调度,线程串行执行。

版权声明:程序员胖胖胖虎阿 发表于 2022年11月25日 上午7:16。
转载请注明:synchronized锁升级 | 胖虎的工具箱-编程导航

相关文章

暂无评论

暂无评论...