synchronized底层原理、锁升级过程解读(带案例)

最近在读Charlie Hunt大神的《Java Performance》,第三章讲《JVM Overview》中间有说到synchronized的一些基本逻辑。本文会做一些整理,主要内容和重要知识点(本文中若未明确说明,JVM默认指的是HotSpot版VM):

  1. synchronized是什么

  2. synchronized有哪些常见用法

  3. synchronized在JVM中的实现原理

    • 同步方法:通过ACC_SYNCHRONIZED标志位来实现
    • 同步代码块:通过monitorentermonitorexit命令来实现
  4. synchronized使用demo和注意点

    • 类对象锁:修饰静态方法和class对象时
    • 实例对象锁:修饰非静态方法、代码块和非class对象时
  5. synchronized锁优化和锁升级过程

    • 无锁
    • 偏向锁
    • 轻量级锁
    • 重量级锁

1. synchronized是什么?

《Java performance》中的定义是:

  Synchronization is described as a mechanism that prevents, avoids, 
  or recovers from the inopportune interleavings, commonly called races, of concurrent operations. 

翻译:

  同步是一种并发操作机制,用来预防、避免对资源不合适的交替使用(通常竞争),保障交替使用资源的安全。

2. synchronized有哪些常见用法

  • 修饰方法
     public static synchronized Integer getAgeOne() { //静态方法
            return age;
        }
     public synchronized Integer getAgeTwo() { //实例方法
          return age;
      }
    
  • 修饰代码块
      public Integer getAgeThree() {
          synchronized (this) {
              return age;
          }
      }
    

3. synchronized在HotSpot VM中的实现原理

  • 方法

    • 通过javap命令反解析class文件,获取synchronized在字节码层面是如何实现的。
  • 步骤

    • 创建一个demo类
    public class SynchronizedDemoOne {
        private static int age = 1;
        /**
         * synchronized 修饰静态方法
         */
        public static synchronized Integer getAgeOne() {
            return age;
        }
        /**
         * synchronized 修饰非静态方法
         */
        public synchronized Integer getAgeTwo() {
            return age;
        }
        /**
         * synchronized 修饰代码块完整
         */
        public Integer getAgeThree() {
            synchronized (this) {
                return age;
            }
        }
    }
    
    • 通过classc命令把java编译成class文件
        javac -g ./SynchronizedDemoOne.java
    
    • 通过classp命令对class文件进行反解析
        javap  -verbose  SynchronizedDemoOne
    
    • 得到反解析后的文件
     Classfile /Users/height/git/learn/JavaAccumulator/src/com/height/concurrent/synchronization/implementation/SynchronizedDemoOne.class
       Last modified 2020-9-9; size 877 bytes
       MD5 checksum bdd02e83e30f0ac316a408694f638868
       Compiled from "SynchronizedDemoOne.java"
     public class com.height.concurrent.synchronization.implementation.SynchronizedDemoOne
       minor version: 0
       major version: 52
       flags: ACC_PUBLIC, ACC_SUPER
     Constant pool:
        #1 = Methodref          #5.#26         
        #2 = Fieldref           #4.#27         
        .                                      //中间省略部分
        .
        .
       #34 = Utf8               valueOf
       #35 = Utf8               (I)Ljava/lang/Integer;
     {
       public com.height.concurrent.synchronization.implementation.SynchronizedDemoOne();
         descriptor: ()V
         flags: ACC_PUBLIC
         Code:
           stack=1, locals=1, args_size=1
              0: aload_0
              1: invokespecial #1                  
              4: return
           LineNumberTable:
             line 3: 0
           LocalVariableTable:
             Start  Length  Slot  Name   Signature
                 0       5     0  this   Lcom/height/concurrent/synchronization/implementation/SynchronizedDemoOne;
     
       public static synchronized java.lang.Integer getAgeOne();
         descriptor: ()Ljava/lang/Integer;
         flags: ACC_PUBLIC, ACC_STATIC, ACC_SYNCHRONIZED   // ACC_SYNCHRONIZED 标志位
         Code:
           stack=1, locals=0, args_size=0
              0: getstatic     #2                  
              3: invokestatic  #3                  
              6: areturn
           LineNumberTable:
             line 8: 0
     
       public synchronized java.lang.Integer getAgeTwo();
         descriptor: ()Ljava/lang/Integer;
         flags: ACC_PUBLIC, ACC_SYNCHRONIZED             // ACC_SYNCHRONIZED 标志位
         Code:
           stack=1, locals=1, args_size=1
              0: getstatic     #2                  
              3: invokestatic  #3                  
              6: areturn
           LineNumberTable:
             line 12: 0
           LocalVariableTable:
             Start  Length  Slot  Name   Signature
                 0       7     0  this   Lcom/height/concurrent/synchronization/implementation/SynchronizedDemoOne;
     
       public java.lang.Integer getAgeThree();
         descriptor: ()Ljava/lang/Integer;
         flags: ACC_PUBLIC
         Code:
           stack=2, locals=3, args_size=1
              0: aload_0
              1: dup
              2: astore_1
              3: monitorenter                             //获取monitor对象
              4: getstatic     #2                  
              7: invokestatic  #3                  
             10: aload_1
             11: monitorexit                              //释放monitor对象
             12: areturn
             13: astore_2
             14: aload_1
             15: monitorexit                              //锁定过程中发生异常时的释放monitor对象
             16: aload_2
             17: athrow
           Exception table:
              from    to  target type
                  4    12    13   any
                 13    16    13   any
           LineNumberTable:
             line 16: 0
             line 17: 4
             line 18: 13
           LocalVariableTable:
             Start  Length  Slot  Name   Signature
                 0      18     0  this   Lcom/height/concurrent/synchronization/implementation/SynchronizedDemoOne;
           StackMapTable: number_of_entries = 1
             frame_type = 255 /* full_frame */
               offset_delta = 13
               locals = [ class com/height/concurrent/synchronization/implementation/SynchronizedDemoOne, class java/lang/Object ]
               stack = [ class java/lang/Throwable ]
            .
            .                                  //省略部分
            .
     }
    
    的文件参见: 反解析完整文件
  • 分析

    • 官方对synchronized关键词的解释是这样的synchronized官方解释

         Method-level synchronization is performed implicitly, as part of method invocation and return. 
         A synchronized method is distinguished in the run-time constant pool’s method_info structure by the ACC_SYNCHRONIZED flag,
         which is checked by the method invocation instructions. When invoking a method for which ACC_SYNCHRONIZED is set, 
         the executing thread enters a monitor, invokes the method itself, and exits the monitor whether the method invocation completes normally or abruptly. 
         During the time the executing thread owns the monitor, no other thread may enter it. 
         If an exception is thrown during invocation of the synchronized method and the synchronized method does not handle the exception, 
         the monitor for the method is automatically exited before the exception is rethrown out of the synchronized method.
      

      翻译下

         同步方法的运行是隐式的,类似于jvm对于方法的引用和返回的支持。同步方法通过在运行常量池里method_info数据结构中的ACC_SYNCHRONIZED标签来标注。
         如果一个线程发现调用的方法有ACC_SYNCHRONIZED标记,那么线程的执行过程就变成:获取monitor对象,调用方法,释放monitor对象。
         在某个线程持有monitor对象时,如果其他线程也想获取该对象,则会别阻塞。
         如果一个同步方法执行过程中发生异常,而且方法自己没有处理,那么在异常被向外抛时,线程也会自动释放monitor对象。
      
    • 官方文档也说的非常清楚了,JVM在处理同步方法时,是通过隐式的获取monitor对象来实现。
      从反解析的class中也可以看到,同步代码块是显式的通过monitor对象来实现互斥访问。
      因此可以简单的归纳下,synchronized关键词的实现,在JVM中,synchronized通过获取monitor对象来实现的。

4. synchronized使用demo和注意点

4.1 案例1
  • 代码
public class SynchronizedDemoTwo {
    public synchronized static void synchronizedStaticMethodMethod() {  //同步静态方法
        System.out.println("synchronized static method start !");
        sleep(1000);
        System.out.println("synchronized static method  end !");
    }
    public static void synchronizedClassMethod() {                     //同步代码块-同步对象为class对象
        synchronized (SynchronizedDemoTwo.class) {
            System.out.println("synchronized class start !");
            sleep(1000);
            System.out.println("synchronized class end !");
        }
    }
    public static void main(String args[]) {
        synchronizedRun();
    }
    private static void synchronizedRun() {
        new Thread(new Runnable() {
            @Override
            public void run() {
                SynchronizedDemoTwo.synchronizedStaticMethodMethod();
            }
        }).start();
        new Thread(new Runnable() {
            @Override
            public void run() {
                SynchronizedDemoTwo.synchronizedClassMethod();
            }
        }).start();
    }
    private static void sleep(int second) {
        try {
            Thread.sleep(second);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}
  • 执行结果
synchronized static method start !          
synchronized static method  end !
synchronized class start !               
synchronized class end !
//在静态同步方法执行结束后才开始执行同步代码块
  • 分析
    • 静态方法和同步参数是class对象时,执行时会获取class对象的锁,所以上述代码会发生锁竞争,执行结果也证实了这个逻辑。
  • 注意点
    • 当你使用synchronized修饰静态方法或者class对象时,要非常谨慎,同一个class只有一把锁,这个锁作用域是非常大的。像String.class,Integer.class这些原生类也不要轻易加锁。
4.2 案例2
  • 代码
public class SynchronizedDemoThree {
    public synchronized void firstSynchronizedMethod() {    //同步方法1
        System.out.println("first synchronized start !");
        sleep(1000);
        System.out.println("first synchronized end !");
    }
    public synchronized void secondSynchronizedMethod() {   //同步方法2
        System.out.println("second synchronized start !");
        sleep(1000);
        System.out.println("second synchronized  end !");
    }
    public void synchronizedBlockMethod() {                //同步代码块-同步对象为实例对象
        synchronized (this) {
            System.out.println("synchronized block start !");
            sleep(1000);
            System.out.println("synchronized block end !");
        }
    }
    public static void main(String args[]) {
        SynchronizedDemoThree demo1 = new SynchronizedDemoThree();
        new Thread(new Runnable() {
            @Override
            public void run() {
                demo1.firstSynchronizedMethod();
            }
        }).start();
        new Thread(new Runnable() {
            @Override
            public void run() {
                demo1.secondSynchronizedMethod();
            }
        }).start();
        new Thread(new Runnable() {
            @Override
            public void run() {
                demo1.synchronizedBlockMethod();
            }
        }).start();
    }
    private static void sleep(int second) {
        try {
            Thread.sleep(second);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}
  • 执行结果
```
first synchronized start !
first synchronized end !
synchronized block start !          
synchronized block end !
second synchronized start !
second synchronized  end !            //有序执行这3个方法,说明发生了竞争
```
  * 分析
    * 实例方法和代码块被synchronized修饰时,执行时会获取实例对象的锁,所以上述代码会发生锁竞争,执行结果也证实了这个逻辑。
  * 注意点
    * 尽量不要用一些公用对象的锁,比如封装类常量池中的一些对象: Integer,String等都有类似的逻辑。

5. synchronized锁优化逻辑和锁升级过程分析

5.1 锁的几种状态

由于开始设计的同步逻辑,在发生互斥资源竞争访问时,等待的线程会变成block状态。而线程的调度是在内核态运行的,所以涉及到了内核态和用户态的切换,而且是2次:block时一次,唤醒时一次。
所以这样的操作效率不高,JDK1.6开始,就对synchronized的机制做了优化,把锁的状态分成了以下几种:

  • 无锁状态:已解锁
  • 偏向锁:已锁定/已解锁且无共享
  • 轻量级锁:已锁定且共享,但非竞争。
  • 重量级锁:已锁定/已解锁且共享和竞争。线程在monitor-enter或wait()时被阻塞。

5.2 举个例子来说明这几种状态

  • 银行交易有一个窗口可以办业务,门口有个取票机和一个引导员(帮助不会操作的客户)。
  • 为了每次只有1个客户到窗口办业务,办业务前客户必须取票,然后系统会根据取票顺序按一定逻辑来叫号。
  • 等待叫号必须去专门的等候区,等候区域距离取票机和窗口都有一定的距离。(系统调度效率不高)
  • 运行一段时间后,发现有时候客户很少,还是需要取号,然后到等候区等待叫号,如果一个同一个客户多次办业务,就需要来回跑。

优化后:

  • 在客户很少的时候,如果窗口空闲,则第一个来办理业务的人,引导员会只需记录他的名字,不用取票,直接让他去办业务,而且只要没有新客户,他多次办业务都不需要取票。(这时候变成了偏向锁)
  • 正在他享受这超级vip服务的时候,又来了新的客户,新客户也知道银行的新规定,没有直接取号,而是询问引导员是否可办业务,引导员说不行,因为现在有人在办。(这时候变成了轻量级锁,通过cas判断是否能获取锁)
  • 新客户知道取票等候区一套流程蛮麻烦,所以告诉引导员说,他可以旁边等一等,前面人办完了,他也想直接进去办业务。(这时候变成了自旋锁)
  • 新客户发现自己询问了10次都没等到办业务,所以直接向银行大堂经理投诉。银行经理就过来说,今天你们不准不取号了,每次进去办业务必须取号。(这时候变成了重锁)

分析:

  • 偏向锁适合的场景是,某段时间内访问互斥资源的线程基本是同一个,没有共享访问的场景
  • 轻量级锁适合的场景是,每次访问互斥资源的时间很短,大家能共享访问,互不影响
  • 重量级锁适合的场景是,常发生竞争,每次占用资源的时间都不短

5.3锁升级简化版

  • Mark Word介绍
    • JVM主要通过对象头中的Mark Word来标记锁的相关状态,包括当前锁的状态和持有锁对象的信息,下面是在不同状态下Mark Word的信息。


      mark word.png
  • 锁升级流程简化版
    • 很多博客中有一个详细版的锁升级流程,我把他们简化了下,更容易理解一些


      锁升级.png
  • 注意点
    • 锁的状态只有4种,无锁->偏向锁->轻量级锁->重量级锁
    • 升级过程不可逆,不同阶段通过从轻到重的方式获取锁
    • 自旋这个操作是通过线程死循环,而防止被阻塞,试图避免用户态和内核态的切换,所以本身不属于锁的状态,是配合轻量级锁使用的一种方式

本文中所有的代码和说明都可以在github中找到,戳这里>


我是大旗,努力用易理解的案例分析进阶知识,一起来学习JVM调优,高并发,常用中间件吧~

如果喜欢我的文章, 来关注我吧~ [岳大旗的博客]

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,445评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,889评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,047评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,760评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,745评论 5 367
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,638评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,011评论 3 398
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,669评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,923评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,655评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,740评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,406评论 4 320
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,995评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,961评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,197评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,023评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,483评论 2 342