如果你在大学中学习过操作系统课程,你可能记得monitor是操作系统中同步的一个重要概念。
它也用于Java同步。这篇文章用一个类比来解释“监控”的基本概念。
一、什么是显示器
监视器可以看作是包含一个特殊房间的建筑物。
这个特殊的房间一次只能容纳一个客户(线程)。
这个房间通常包含一些数据和代码。
如果客户想要占用这个特殊的房间,他必须先进入走廊(入口)等待。
调度程序将选择一个基于一些标准(例如。先进先出)。
如果他因为某种原因被停职,他将被送到等候室,并安排稍后重新进入特别室。
如上图所示,这栋楼有三个房间。
简单地说,监视器是监视线程对特定房间的访问的工具。
它确保只有一个线程可以访问受保护的数据或代码。
二、它在Java中是如何实现的
在Java虚拟机中,每个对象和类在逻辑上都与监视器相关联。
为了实现监视器的互斥功能,锁(有时称为互斥锁)与每个对象和类相关联。
这在操作系统书籍中叫做信号量,互斥锁是二进制信号量。
如果一个线程拥有某些数据上的锁,那么在拥有锁的线程释放锁之前,其他线程无法获得该锁。
如果我们在进行多线程编程时需要一直编写信号量,那就不太方便了。
幸运的是,我们不需要这样做,因为JVM会自动为我们这样做。
要声明一个监视区域(这意味着数据不能被多个线程访问),Java提供了同步语句和同步方法。
一旦代码嵌入synchronized关键字,它就是一个监控区域。
锁是由JVM在后台自动实现的。
三、在Java同步代码中,监视器是哪一部分
我们知道每个对象/类都与一个监视器相关联。
我认为这样说很好,每个对象都有一个监视器,因为每个对象都可以有自己的临界区,并且能够监视线程序列。
为了支持不同线程之间的协作,Java提供了wait()和notify()来分别挂起一个线程和唤醒另一个在对象上等待的线程。
另外还有3个版本:
这些方法只能在同步语句或同步方法中调用。
原因是如果一个方法不需要互斥,那么就不需要在线程之间监视或协作,每个线程都可以自由地访问该方法。