Object类wait,notify,notifyAll的使用

本文详细介绍了Java基础类Object中的wait(), notify(), notifyAll()方法的使用。这些方法用于线程间的同步,确保线程安全地访问共享资源。文章通过实例解释了如何正确使用这些方法,避免出现IllegalMonitorStateException异常。

这三个方法是java的基础类Object中定义的。

 

wait()

JDk文档写道
在其他线程调用此对象的 notify() 方法或 notifyAll() 方法前,导致当前线程等待。换句话说,此方法的行为就好像它仅执行 wait(0) 调用一样。
当前线程必须拥有此对象监视器 。该线程发布对此监视器的所有权并等待,直到其他线程通过调用 notify 方法,或 notifyAll 方法通知在此对象的监视器上等待的线程醒来。然后该线程将等到重新获得对监视器的所有权后才能继续执行。

对于某一个参数的版本,实现中断和虚假唤醒是可能的,而且此方法应始终在循环中使用:

synchronized (obj) {
while (<condition does not hold>)
obj.wait();
... // Perform action appropriate to condition
}
此方法只应由作为此对象监视器的所有者的线程来调用。
抛出:
IllegalMonitorStateException - 如果当前线程不是此对象监视器的所有者。
InterruptedException - 如果在当前线程等待通知之前或者正在等待通知时,任何线程中断了当前线程。在抛出此异常时,当前线程的 中断状态 被清除。

 对于红色部分的内容,个人曾一直都不是很理解,什么叫做拥有此对象的监视器。下面我们看看代码:

 

	DateFormat format = new SimpleDateFormat("yyyy-MM-dd:hh:mm:ss");
	
	private String getTime(){
		return format.format(Calendar.getInstance().getTime());
	}
	
	private Object monitor = new Object();


        public void waitOnce(String thread, final long ms) {
		Thread waitThread = new Thread() {
			public void run() {
				synchronized (monitor) {//获得对象监视器

					try {
						System.out.println("Thread "
								+ Thread.currentThread().getName()
								+ " Wait at " + getTime());
						monitor.wait(ms);
						System.out.println("Thread "
								+ Thread.currentThread().getName()
								+ " Waked at " + getTime());
					} catch (InterruptedException e) {
					}
				}
			};
		};
		waitThread.setName(thread);
		waitThread.start();
	}

 

如果我们去掉synchronized(monitor) ,运行则会出现异常IllegalMonitorStateException。

 

		WaitAndNotifyTest test = new WaitAndNotifyTest();
		test.waitOnce("1", Long.MAX_VALUE);
写道
Exception in thread "1" java.lang.IllegalMonitorStateException

 而加上以后就没问题了。因此个人觉得使用synchronized关键字锁定对象,也就是获得了对象的监视器了。

 

notify()

JDK文档 写道
唤醒在此对象监视器上等待的单个线程。如果所有线程都在此对象上等待,则 会选择唤醒其中一个线程 。选择是 任意性 的,并在对实现做出决定时发生。线程通过调用其中一个 wait 方法,在对象的监视器上等待。

直到当前线程放弃此对象上的锁定,才能继续执行被唤醒的线程 。被唤醒的线程将以常规方式与在该对象上主动同步的其他所有线程进行竞争;例如,唤醒的线程在作为锁定此对象的下一个线程方面没有可靠的特权或劣势。

此方法只应由作为此对象监视器的所有者的线程来调用。通过以下三种方法之一,线程可以成为此对象监视器的所有者:

通过执行此对象的同步实例方法。
通过执行在此对象上进行同步的 synchronized 语句的正文。
对于 Class 类型的对象,可以通过执行该类的同步静态方法。


一次只能有一个线程拥有对象的监视器。

抛出:
IllegalMonitorStateException - 如果当前线程不是此对象监视器的所有者。

 首先理解一下获得对象的监视器,简单的说就是取得了当前对象的“加锁”使用权,最简单的就是使用synchronized关键字。另外使用 synchronized修饰的方法也行。

 

notify方法还有一个值得提出的是它会在当前线程释放了对象锁以后随机唤醒一个在该对象上等待的线程

看看一个例子:

	public void awakeAndWait(String thread, final long ms) {
		Thread notifyThread = new Thread() {
			public void run() {
				synchronized (monitor) {
					monitor.notify();
					System.out.println("Thread "
							+ Thread.currentThread().getName() + " Notify at "
							+ getTime());
					//保持了对象锁的等待
					try {
						Thread.sleep(ms);
					} catch (InterruptedException e) {
					}
				}
				//释放了对象锁之后的等待
				try {
					Thread.sleep(ms);
				} catch (InterruptedException e) {
				}
			};
		};
		notifyThread.setName(thread);
		notifyThread.start();
	}

 这个方法会唤醒一个在对象上等待的线程,并在两次sleep后退出,注意的是一个sleep是在对象锁内,而另一次则是在释放了对象锁以后,这时候运行上面2个方法得到:

		WaitAndNotifyTest test = new WaitAndNotifyTest();
		test.waitOnce("1", Long.MAX_VALUE);// 在对象上等待无限长
		test.waitOnce("2", Long.MAX_VALUE);// 在对象上等待无限长
		test.waitOnce("3", Long.MAX_VALUE);// 在对象上等待无限长
		try {// 延迟2s
			Thread.sleep(2000);
		} catch (InterruptedException e) {
		}
		// 在唤醒一个在对象上等待的线程,本身执行时间4s,2s是在对象锁内
		//,2s是在释放了对象锁以后
		test.awakeAndWait("3", 2000);

 执行结果为:

写道
Thread 1 Wait at 2011-05-06:10:57:04
Thread 2 Wait at 2011-05-06:10:57:04
Thread 3 Wait at 2011-05-06:10:57:04
Thread 3 Notify at 2011-05-06:10:57:06
Thread 1 Waked at 2011-05-06:10:57:08

 2秒后唤醒了线程1,尽管它自己执行花了4s,在释放了对象锁之后的2s不会影响线程1的执行。

 

notifyAll()

JDK文档 写道
唤醒在此对象监视器上等待的 所有线程 。线程通过调用其中一个 wait 方法,在对象的监视器上等待。
直到当前线程放弃此对象上的锁定,才能继续执行被唤醒的线程。被唤醒的线程将以常规方式与在该对象上主动同步的其他所有线程进行竞争;例如,唤醒的线程在作为锁定此对象的下一个线程方面没有可靠的特权或劣势。

此方法只应由作为此对象监视器的所有者的线程来调用。有关线程能够成为监视器所有者的方法的描述,请参阅 notify 方法。

抛出:
IllegalMonitorStateException - 如果当前线程不是此对象监视器的所有者。

 与notify稍微有一点差别的是,它会唤醒所有的等待线程。

 

	public void awakeAll(String thread) {
		Thread notifyThread = new Thread() {
			public void run() {
				synchronized (monitor) {
					monitor.notifyAll();
					System.out.println("Thread "
							+ Thread.currentThread().getName()
							+ " Notify all at " + getTime());
				}
			};
		};
		notifyThread.setName(thread);
		notifyThread.start();
	}

 执行一下代码:

		WaitAndNotifyTest test = new WaitAndNotifyTest();
		test.waitOnce("1", Long.MAX_VALUE);// 在对象上等待无限长
		test.waitOnce("2", Long.MAX_VALUE);// 在对象上等待无限长
		test.waitOnce("3", Long.MAX_VALUE);// 在对象上等待无限长
		try {// 延迟2s
			Thread.sleep(2000);
		} catch (InterruptedException e) {
		}
		test.awakeAll("4");

 结果为:

写道
Thread 1 Wait at 2011-05-06:10:59:15
Thread 3 Wait at 2011-05-06:10:59:15
Thread 2 Wait at 2011-05-06:10:59:15
Thread 4 Notify all at 2011-05-06:10:59:17
Thread 2 Waked at 2011-05-06:10:59:17
Thread 3 Waked at 2011-05-06:10:59:17
Thread 1 Waked at 2011-05-06:10:59:17

 全部唤醒了。

 

总结

总结一下:大概有以下几点:

  1. wait(),notify(),notifyAll()都需要在拥有对象监视器的前提下执行,否则会出现异常IllegalMonitorStateException。
  2. 多个线程可以同时在一个对象上等待。
  3. notify()将随机唤醒一个在对象上等待的线程,没有一个都没有,则什么都不做。
  4. notify()唤醒的线程,将在notify()线程释放了对象监视器以后才执行,并不是notify了以后马上执行。
  5. Object的这些方法与Thread的sleep、interrupt相差还是很远的,不要混为一谈了。
欢迎使用“可调增益放大器 Multisim”设计资源包!本资源专为电子爱好者、学生以及工程师设计,旨在展示如何在著名的电路仿真软件Multisim环境下,实现一个具有创新性的数字控制增益放大器项目。 项目概述 在这个项目中,我们通过巧妙结合模拟电路与数字逻辑,设计出一款独特且实用的放大器。该放大器的特点在于其增益可以被精确调控,并非固定不变。用户可以通过控制键,轻松地改变放大器的增益状态,使其在1到8倍之间平滑切换。每一步增益的变化都直观地通过LED数码管显示出来,为观察和调试提供了极大的便利。 技术特点 数字控制: 使用数字输入来调整模拟放大器的增益,展示了数字信号对模拟电路控制的应用。 动态增益调整: 放大器支持8级增益调节(1x至8x),满足不同应用场景的需求。 可视化的增益指示: 利用LED数码管实时显示当前的放大倍数,增强项目的交互性和实用性。 Multisim仿真环境: 所有设计均在Multisim中完成,确保了设计的仿真准确性和学习的便捷性。 使用指南 软件准备: 确保您的计算机上已安装最新版本的Multisim软件。 打开项目: 导入提供的Multisim项目文件,开始查看或修改设计。 仿真体验: 在仿真模式下测试放大器的功能,观察增益变化及LED显示是否符合预期。 实验与调整: 根据需要调整电路参数以优化性能。 实物搭建 (选做): 参考设计图,在真实硬件上复现实验。
在Java并发编程中,`wait()`, `notify()`, 和 `notifyAll()` 是 `Object` 的一部分,用于线程间的通信,特别是在多线程同步中。它们主要用于 `synchronized` 关键字修饰的代码块或方法中。 1. `wait()`: 当一个线程持有某个对象的锁并且想要等待特定条件时,可以调用 `wait()`。这将释放锁,并让当前线程进入等待状态,直到有其他线程通过调用 `notify()` 或 `notifyAll()` 将其唤醒。在此期间,线程不会执行,而是处于不可中断的阻塞状态。 2. `notify()`: 只有一个线程能够被唤醒,即通知第一个正在等待的线程,使其恢复执行。如果你只想唤醒一个等待的线程,应该使用此方法。 3. `notifyAll()`: 另一方面,`notifyAll()` 会唤醒所有等待在该对象上的线程。这意味着如果有多个线程在同一个对象上等待,调用 `notifyAll()` 可能会使所有线程都恢复执行。 示例代码: ```java class SharedResource { private final Object lock = new Object(); private boolean resourceAvailable = false; synchronized void acquireResource() throws InterruptedException { while (!resourceAvailable) { wait(); // 等待资源可用 } resourceAvailable = false; // 表明资源已使用 } synchronized void releaseResource() { resourceAvailable = true; // 设置资源可用 notifyAll(); // 唤醒所有等待的线程 } } // 在其他线程中调用 SharedResource shared = new SharedResource(); shared.acquireResource(); // ...使用资源... shared.releaseResource(); ```
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值