Synchronized简介

Synchronized的正确用法

  • 修饰实例方法

    /**
     * @Author: zenghzong
     * @Description:
     * @Date: Create in 16:18 2020/5/18
     */
    public class Main {
        public synchronized void test(){
        }
    }
    
  • 修饰静态方法

    public class Main {
        public  void test(){
            synchronized(Main.class){
    
            }
        }
    }
    
  • 修饰代码块

    public class Main {
        public void test1(){
            synchronized(new test2()){
    
            }
        }
    }
    

    synchronized底层实现

    java源代码

    /**
     * @Author: zenghzong
     * @Description:
     * @Date: Create in 16:18 2020/5/18
     */
    public class Main {
        public synchronized void test(){
            synchronized(new String()){
    
            }
        }
    }
    

    编译之后字节码,这里的对象头,他会关联到一个monitor对象。

    Classfile /D:/myproject/Main.class
      Last modified 2020-5-18; size 490 bytes
      MD5 checksum 049256e45d8bd6241ed186ac84a10f8e
      Compiled from "Main.java"
    public class io.renren.modules.sys.Main
      minor version: 0
      major version: 52
      flags: ACC_PUBLIC, ACC_SUPER
    Constant pool:
       #1 = Methodref          #5.#20         // java/lang/Object."<init>":()V
       #2 = Class              #21            // java/lang/String
       #3 = Methodref          #2.#20         // java/lang/String."<init>":()V
       #4 = Class              #22            // io/renren/modules/sys/Main
       #5 = Class              #23            // java/lang/Object
       #6 = Utf8               <init>
       #7 = Utf8               ()V
       #8 = Utf8               Code
       #9 = Utf8               LineNumberTable
      #10 = Utf8               LocalVariableTable
      #11 = Utf8               this
      #12 = Utf8               Lio/renren/modules/sys/Main;
      #13 = Utf8               test
      #14 = Utf8               StackMapTable
      #15 = Class              #22            // io/renren/modules/sys/Main
      #16 = Class              #23            // java/lang/Object
      #17 = Class              #24            // java/lang/Throwable
      #18 = Utf8               SourceFile
      #19 = Utf8               Main.java
      #20 = NameAndType        #6:#7          // "<init>":()V
      #21 = Utf8               java/lang/String
      #22 = Utf8               io/renren/modules/sys/Main
      #23 = Utf8               java/lang/Object
      #24 = Utf8               java/lang/Throwable
    {
      public io.renren.modules.sys.Main();
        descriptor: ()V
        flags: ACC_PUBLIC
        Code:
          stack=1, locals=1, args_size=1
             0: aload_0
             1: invokespecial #1                  // Method java/lang/Object."<init>":()V
             4: return
          LineNumberTable:
            line 8: 0
          LocalVariableTable:
            Start  Length  Slot  Name   Signature
                0       5     0  this   Lio/renren/modules/sys/Main;
    
      public synchronized void test();
        descriptor: ()V
        flags: ACC_PUBLIC, ACC_SYNCHRONIZED
        Code:
          stack=2, locals=3, args_size=1
             0: new           #2                  // class java/lang/String
             3: dup
             4: invokespecial #3                  // Method java/lang/String."<init>":()V
             7: dup
             8: astore_1
             9: monitorenter
            10: aload_1
            11: monitorexit
            12: goto          20
            15: astore_2
            16: aload_1
            17: monitorexit
            18: aload_2
            19: athrow
            20: return
          Exception table:
             from    to  target type
                10    12    15   any
                15    18    15   any
          LineNumberTable:
            line 10: 0
            line 12: 10
            line 13: 20
          LocalVariableTable:
            Start  Length  Slot  Name   Signature
                0      21     0  this   Lio/renren/modules/sys/Main;
          StackMapTable: number_of_entries = 2
            frame_type = 255 /* full_frame */
              offset_delta = 15
              locals = [ class io/renren/modules/sys/Main, class java/lang/Object ]
              stack = [ class java/lang/Throwable ]
            frame_type = 250 /* chop */
              offset_delta = 4
    }
    SourceFile: "Main.java"
    

    进入一个方法的时候,执行monitorenter,获得对象的所以权,如果monitor进入数为1,当前的这个线程就是这个monitorowner,如果这时候已经是这个monitorowner了,再次进入,就会更新技术成num+1。如果执行完monitorexit,num就-1,直到为0,然后才可以被其他线程持有。加锁的互斥,就是看你能否获得monitor的使用权,一旦你成为owner就是获得者。

    monitor

    monitor源码是C++写的,在虚拟机的ObjectMonitor.hpp文件中。该文件在线地址,该对象的结构如下

在这里插入图片描述

    _recursions   = 0;  // 线程重入次数
    _object       = NULL;  // 存储Monitor对象
    _owner        = NULL;  // 持有当前线程的owner
    _WaitSet      = NULL;  // wait状态的线程列表
 	 _cxq          = NULL ;  // 单向列表
    _EntryList    = NULL ;  // 处于等待锁状态block状态的线程列表

synchronized底层的源码就是引入了ObjectMonitor

TODO monitor为什么两次exit,一次enter

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

技术宅丶拾年

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值