Java Monitor

什么是Monitor

Monitor可以理解为一种同步工具,也可理解为一种同步机制,常常被描述为一个Java对象,也叫管程

管程(Monitor)是一种和信号量(Sophomore)等价的同步机制。它在Java并发编程中也非常重要,虽然程序员没有直接接触管程,但它确实是synchronized和wait()/notify()等线程同步和线程间协作工具的基石:当我们在使用这些工具时,其实是它在背后提供了支持。简单来说:管程使用锁(lock)确保了在任何情况下管程中只有一个活跃的线程,即确保线程互斥访问临界区管程使用条件变量(Condition Variable)提供的等待队列(Waiting Set)实现线程间协作,当线程暂时不能获得所需资源时,进入队列等待,当线程可以获得所需资源时,从等待队列中唤醒。

总结:

  • 互斥:一个Monitor在一个时刻只能被一个线程持有,即Monitor中的所有方法都是互斥的。

  • signal机制:如果条件变量不满足,允许一个正在持有Monitor的线程暂时释放持有权,当条件变量满足时,当前线程可以唤醒正在等待该条件变量的线程,然后重新获取Monitor的持有权。

所有的Java对象是天生的Monitor,每一个Java对象都有成为Monitor的潜质,因为在Java的设计中 ,每一个Java对象自打娘胎里出来就带了一把看不见的锁,它叫做内部锁或者Monitor锁。

Monitor的本质是依赖于底层操作系统的Mutex Lock实现,操作系统实现线程之间的切换需要从用户态到内核态的转换,成本非常高。

Monitor 是线程私有的数据结构,每一个线程都有一个可用monitor record列表,同时还有一个全局的可用列表。每一个被锁住的对象都会和一个monitor关联(对象头的MarkWord中的LockWord指向monitor的起始地址),同时monitor中有一个Owner字段存放拥有该锁的线程的唯一标识,表示该锁被这个线程占用。其结构如下:

  • Owner字段:初始时为NULL表示当前没有任何线程拥有该monitor record,当线程成功拥有该锁后保存线程唯一标识,当锁被释放时又设置为NULL
  • EntryQ字段:关联一个系统互斥锁(semaphore),阻塞所有试图锁住monitor record失败的线程
  • RcThis字段:表示blocked或waiting在该monitor record上的所有线程的个数
  • Nest字段:用来实现重入锁的计数
  • HashCode字段:保存从对象头拷贝过来的HashCode值(可能还包含GC age)
  • Candidate字段:用来避免不必要的阻塞或等待线程唤醒,因为每一次只有一个线程能够成功拥有锁,如果每次前一个释放锁的线程唤醒所有正在阻塞或等待的线程,会引起不必要的上下文切换(从阻塞到就绪然后因为竞争锁失败又被阻塞)从而导致性能严重下降;Candidate只有两种可能的值0表示没有需要唤醒的线程1表示要唤醒一个继任线程来竞争锁

Monitor具体实现方式

  1. Monitor是在jvm底层实现的,底层代码是c++
  2. Monitor的enter方法:获取锁
  3. Monitor的exit方法:释放锁
  4. Monitor的wait方法:为java的Object的wait方法提供支持
  5. Monitor的notify方法:为java的Object的notify方法提供支持
  6. Monitor的notifyAll方法:为java的Object的notifyAll方法提供支持

Monitor机制

见下图:

Monitor可以类比为一个特殊的房间,这个房间中有一些被保护的数据,Monitor保证每次只能有一个线程能进入这个房间进行访问被保护的数据,进入房间即为持有Monitor,退出房间即为释放Monitor。

当一个线程需要访问受保护的数据(即需要获取对象的Monitor)时,它会首先在entry-set入口队列中排队(这里并不是真正的按照排队顺序),如果没有其他线程正在持有对象的Monitor,那么它会和entry-set队列和wait-set队列中的被唤醒的其他线程进行竞争(即通过CPU调度),选出一个线程来获取对象的Monitor,执行受保护的代码段,执行完毕后释放Monitor,如果已经有线程持有对象的Monitor,那么需要等待其释放Monitor后再进行竞争。

再说一下wait-set队列。当一个线程拥有Monitor后,经过某些条件的判断(比如用户取钱发现账户没钱),这个时候需要调用Object的wait方法,线程就释放了Monitor,进入wait-set队列,等待Object的notify方法(比如用户向账户里面存钱)。当该对象调用了notify方法或者notifyAll方法后,wait-set中的线程就会被唤醒,然后在wait-set队列中被唤醒的线程和entry-set队列中的线程一起通过CPU调度来竞争对象的Monitor,最终只有一个线程能获取对象的Monitor。

注意:
当一个线程在wait-set中被唤醒后,并不一定会立刻获取Monitor,它需要和其他线程去竞争
如果一个线程是从wait-set队列中唤醒后,获取到的Monitor,它会去读取它自己保存的PC计数器中的地址,从它调用wait方法的地方开始执行。
2.3、Monitor与java对象以及线程是如何关联
1.如果一个java对象被某个线程锁住,则该java对象的Mark Word字段中LockWord指向monitor的起始地址
2.Monitor的Owner字段会存放拥有相关联对象锁的线程id

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Java的对象头是对象实例在内存的一部分,包含了用于控制对象状态的信息。对象头的内容包括标记字、类型指针、数组长度和锁信息等。标记字用于表示对象是否被锁定、是否已经被回收等状态信息。类型指针则指向对象的类元数据信息,包括类的类型、字段信息和方法信息等。数组长度记录了数组对象的长度信息。锁信息包括了对象的同步状态,用于实现Java的同步机制。通过监控对象头,我们可以了解对象在内存的状态信息,为性能分析和问题排查提供帮助。 ### 回答2: Javamonitor对象头是用于实现线程同步和互斥的一种机制。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、付费专栏及课程。

余额充值