Monitor对象是什么?

synchronized保证线程同步的作用相信大家都已经非常熟悉了,可以把任意一个对象当作锁。synchronized 关键字无论是修饰代码块,还是修饰实例方法和静态方法,本质上都是作用于对象上

多个线程要竞争共享资源,而操作共享资源资源的代码就在临界区内,想要进入到这个临界区就必须持有锁

当用 synchronized 修饰代码块时,编译后的字节码会有 monitorentermonitorexit 指令,分别对应的是获得锁和解锁。

当用 synchronized 修饰方法时,会给方法加上标记 ACC_SYNCHRONIZED,这样 JVM 就知道这个方法是一个同步方法,于是在进入同步方法的时候就会进行执行竞争锁的操作,只有拿到锁才能继续执行。

对象锁长啥样?

那么对象锁在内存中是怎样的呢?接下来就来看一下对象锁的实现细节。对象锁的状态是记录在对象头中的Mark word区域中。关于对象的内存区域的细节,大家可以参考前文《Java面试必考问题:对象在内存中是如何布局的? 》
在这里插入图片描述

在不同的锁状态下,Mark word会存储不同的信息,这也是为了节约内存常用的设计。当锁状态为重量级锁(锁标识位=10)时,Mark word中会记录指向Monitor对象的指针,这个Monitor对象也称为管程或监视器锁
在这里插入图片描述

每个对象都存在着一个 Monitor对象与之关联。执行 monitorenter 指令就是线程试图去获取 Monitor 的所有权,抢到了就是成功获取锁了;执行 monitorexit 指令则是释放了Monitor的所有权

ObjectMonitor类

在HotSpot虚拟机中,Monitor是基于C++ObjectMonitor类实现的,其主要成员包括:

  • _owner:指向持有ObjectMonitor对象的线程
  • _WaitSet:存放处于wait状态的线程队列,即调用wait()方法的线程
  • _EntryList:存放处于等待锁block状态的线程队列
  • _count:约为_WaitSet 和 _EntryList 的节点数之和
  • _cxq: 多个线程争抢锁,会先存入这个单向链表
  • _recursions: 记录重入次数

在这里插入图片描述

上图简略展示了ObjectMonitor的基本工作机制:

(1)当多个线程同时访问一段同步代码时,首先会进入 _EntryList 队列中。

(2)当某个线程获取到对象的Monitor后进入临界区域,并把Monitor中的 _owner 变量设置为当前线程,同时Monitor中的计数器 _count 加1。即获得对象锁。

(3)若持有Monitor的线程调用 wait() 方法,将释放当前持有的Monitor,_owner变量恢复为null,_count自减1,同时该线程进入 _WaitSet 集合中等待被唤醒。

(4)在_WaitSet 集合中的线程会被再次放到_EntryList 队列中,重新竞争获取锁。

(5)若当前线程执行完毕也将释放Monitor并复位变量的值,以便其他线程进入获取锁。

线程争抢锁的过程要比上面展示得更加复杂。除了_EntryList 这个双向链表用来保存竞争的线程,ObjectMonitor中还有另外一个单向链表 _cxq,由两个队列来共同管理并发的线程。

在这里插入图片描述
ObjectMonitor::enter() 和 ObjectMonitor::exit() 分别是ObjectMonitor获取锁和释放锁的方法。线程解锁后还会唤醒之前等待的线程,根据策略选择直接唤醒_cxq队列中的头部线程去竞争,或者将_cxq队列中的线程加入_EntryList,然后再唤醒_EntryList队列中的线程去竞争。

ObjectMonitor::enter()
在这里插入图片描述
下面我们看一下ObjectMonitor::enter()方法竞争锁的流程:

首先尝试通过 CAS 把 ObjectMonitor 中的 _owner 设置为当前线程,设置成功就表示获取锁成功。通过 _recursions 的自增来表示重入。

如果没有CAS成功,那么就开始启动自适应自旋,自旋还不行的话,就包装成 ObjectWaiter 对象加入到 _cxq 单向链表之中。关于自旋锁和自适应自旋,可以参考前文《Java面试必考问题:什么是自旋锁 》](https://www.toutiao.com/i6934327407897854475/?group_id=6934327407897854475)。

加入_cxq链表后,再次尝试是否可以CAS拿到锁,再次失败就要阻塞(block),底层调用了pthread_mutex_lock

ObjectMonitor::exit()方法

线程执行 Object.wait()方法时,会将当前线程加入到 _waitSet 这个双向链表中,然后再运行ObjectMonitor::exit() 方法来释放锁。

可重入锁就是根据 _recursions 来判断的,重入一次就执行 _recursions++,解锁一次就执行 _recursions–,如果 _recursions 减到 0 ,就说明需要释放锁了。

线程解锁后还会唤醒之前等待的线程。当线程执行 Object.notify()方法时,从 _waitSet 头部拿线程节点,然后根据策略(QMode指定)决定将线程节点放在哪里,包括_cxq 或 _EntryList 的头部或者尾部,然后唤醒队列中的线程。

在这里插入图片描述

  • 6
    点赞
  • 41
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Java中的对象头是对象实例在内存中的一部分,包含了用于控制对象状态的信息。对象头的内容包括标记字、类型指针、数组长度和锁信息等。标记字用于表示对象是否被锁定、是否已经被回收等状态信息。类型指针则指向对象的类元数据信息,包括类的类型、字段信息和方法信息等。数组长度记录了数组对象的长度信息。锁信息包括了对象的同步状态,用于实现Java中的同步机制。通过监控对象头,我们可以了解对象在内存中的状态信息,为性能分析和问题排查提供帮助。 ### 回答2: Java中的monitor对象头是用于实现线程同步和互斥的一种机制。Monitor对象头实际上是存在于每个Java对象对象头中的一个标志位,用来标示该对象是否被锁定,以及锁的相关信息。 在Java虚拟机中,每个对象对象头占据8字节的空间,其中2字节存储monitor对象头,用于记录锁的状态和计数器等信息。在Java的并发编程中,对一个对象进行同步的操作就是通过获取和释放这个monitor对象来实现的。 当一个线程想要获取一个对象的锁时,它首先会检查该对象monitor对象头是否被其他线程锁定。如果对象没有被锁定,当前线程就会尝试获取这个锁,并将monitor对象头标记为被当前线程所拥有。此时,其他任何线程都无法获取该对象的锁,直到当前线程释放了这个锁。 如果一个对象已经被锁定,那么当前线程就会进入到对象的等待队列中,直到锁被释放。同时,这个monitor对象头会记录锁的计数器,用于区分同一个线程多次获取锁的次数。只有当锁的计数器归零时,锁才会完全释放,其他线程才能够获取到这个对象的锁。 监视器对象头在Java多线程编程中起着非常重要的作用,它提供了线程间的互斥和同步机制,使得多个线程可以安全地共享对象。借助于monitor对象头,可以有效地避免多个线程同时访问共享资源导致的并发问题,从而保证程序的正确性和安全性。 ### 回答3: Java 中的 monitor 对象头是指每个对象在内存中的一块特定区域,用于存储对象的元数据信息和同步状态。它包含了以下几个重要的字段: 1. Mark Word(标记字段):用于存储对象的哈希码、锁状态、GC 分代年龄、锁标志等信息。它是对象的标志,用于判断对象的锁状态。 2. Klass Pointer(类型指针):指向对象的类元数据信息,包括类的方法、字段、父类、实现的接口等信息。通过这个指针,可以得到对象的具体类型以及相关的类信息。 3. Array Length(数组长度):仅在表示数组对象时使用,用于存储数组的长度。 4. Instance Data(实例数据):用于存储对象的实例变量。根据对象的类型,实例数据的存储方式可能会有所不同。 5. Padding(填充字段):为了对象在内存中对齐而添加的填充字段,保证对象的内存布局符合 JVM 的要求。 monitor 对象头的主要作用是管理对象的同步状态,提供了对对象的并发访问控制。在 Java 中,每个对象都与一个 monitor 相关联,它用于实现 synchronized 关键字的功能。当一个线程试图获取对象的锁时,monitor 对象的锁标志会被设置为锁定状态,其他线程无法获取该对象的锁,必须等待锁的释放。 除了锁机制,monitor 对象头还用于实现线程的等待/通知机制。通过调用 Object 类中的 wait()、notify() 和 notifyAll() 方法,线程可以进入等待状态并释放对象的锁,其他线程可以通过 notify() 或 notifyAll() 方法来唤醒等待的线程。 总之,Java 中的 monitor 对象头是用于存储对象的元数据信息和同步状态的特定区域,它在多线程环境中实现了线程间的同步和互斥。它是实现 synchronized 关键字以及等待/通知机制的基础。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值