线程间通信——等待(wait)和通知(notify)

本文介绍了Java中线程间通信的关键方法wait和notify,详细讲解了它们的执行机制、为何wait时必须释放锁以及为何需与synchronized配合使用。wait会导致线程释放对象锁进入等待队列,而notify会随机唤醒一个等待的线程尝试重新获取锁。正确使用wait和notify是确保多线程间有效协作的关键。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1:等待(wait)和通知(notify)的介绍

JDK提供了两个非常重要的接口线程:等待wait()方法和通知notify()方法。这两个方法在Object类中,意味着任何对象都可以调用这两个方法。方法签名如下

public final void wait() throws InterruptedException {
   
public final native void notify();

当在一个对象实例上调用wait方法后,当前线程就会在这个对象上等待。这是什么意思呢?比如,在线程A中,调用了obj.wait()方法,那么线程A就会停止继续执行,转为等待状态。等待到何时结束呢?线程A会一直等到其他线程调用了obj.notify()方法为止。这时,object对象成为了多个线程之间通信的有效手段

2:等待(wait)和通知(notify)的执行机制

那么waitO方法和notify0方法究竟是如何工作的呢?下图展示了两者的工作过程。如果一个线程调用了object.wait()方法,那么它就会进入object对象的等待队列。这个等待队列中,可能会有多个线程,因为系统运行多个线程同时等待某一个对象。 当objec.notify()方法被调用时,它就会从这个等待队列中随机选择一个线程, 并将其唤醒。这个选择是不公平的,并不是先等待的线程就会优先被选择,这个选择完全是随机的。
在这里插入图片描述

除notifyO方法外,Object 对象还有一个类似的notifyAll()方法,它和notif()方法的功能基本一致,不同的是,它会唤醒在这个等待队列中所有等待的线程,而不是随机选择一个。

这里还需要强调一点,Object.waitO方法 并不能随便调用。它必须包含在对应的synchronzied语句中,无论是wait(0方法或者notify0方法都需要首先获得目标对象的一个监视器。下图 显示了wait()方法和notify()方法的工作流程细节。其中T1和T2表示两个线程。TI在正确执行wait方法前,必须获得object对象的监视器。而wait方法在执行后,会释放这个监视器。这样做的目的是使其他等待在object对象上的线程不至于因为T1的休眠而全部无法正常执行。

线程T2在notify方法调用前,也必须获得objet对象的监视器。所幸,此时T1已经释放了这个监视器。因此,T2可以顺利获得object对象的监视器。接着,T2执行了notify方法尝试唤醒一个等待线程,这里假设唤醒了T1。 T1在被唤醒后,要做的第一件事并不是执行后续的代码,而是要尝试重新获得object 对象的监视器,而这个监视器也正是T1在wait方法执行前所持有的那个。如果暂时无法获得,则T1还必须等待这个监视器。当监视器顺利获得后,T1才可以真正意义上继续执行

在这里插入图片描述
这里给出一个使用wait和notify方法的简单案例

public class ThreadWaitAndNotify {
   
    static Object object=new Object()
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值