商城首页欢迎来到中国正版软件门户

您的位置:首页 >怎么通过 Object 类的 wait/notify 机制在面向对象层面实现初级的线程间协作

怎么通过 Object 类的 wait/notify 机制在面向对象层面实现初级的线程间协作

  发布于2026-05-03 阅读(0)

扫一扫,手机访问

怎么通过 Object 类的 wait/notify 机制在面向对象层面实现初级的线程间协作

怎么通过 Object 类的 wait/notify 机制在面向对象层面实现初级的线程间协作

在Ja va多线程编程里,让线程之间有序协作,而不是乱成一锅粥,是个基本功。Object 类提供的 wait()notify()(以及它的兄弟 notifyAll())方法,正是实现这种“等待-唤醒”契约的核心工具。简单来说,就是让一个线程在条件不满足时先“歇着”,等另一个线程准备好后再把它“叫醒”。

但这里有个铁律,必须牢记:所有对 waitnotify 的调用,都必须老老实实地放在 synchronized 同步块里,而且锁对象,必须就是调用这些方法的那个对象本身。这是整个机制能安全运转的基石。

共享状态对象作为协作枢纽

怎么理解这个“契约”呢?关键在于,要有一个共享的状态对象。你可以定义一个普通的Ja va类,比如叫 TaskQueue,用它来封装需要协作的业务状态——比如任务列表是不是空了,或者计算结果有没有准备好。这个对象身兼三职:它是数据的载体,是同步用的锁,更是线程间通信的“信号灯”。

  • 线程A想调用 queue.wait() 去等待,前提是它必须先拿到 queue 这把锁(也就是进入 synchronized(queue) 块)。一旦调用 wait(),它就会释放锁,然后乖乖进入等待队列,直到被唤醒。
  • 线程B在修改了共享状态(比如往队列里加了新任务)之后,同样需要在 synchronized(queue) 块里,调用 queue.notify()queue.notifyAll() 来发出信号。
  • 被唤醒的线程(比如刚才的线程A)会重新加入竞争,去抢 queue 的锁。抢到之后,它才能从 wait() 的地方继续执行。这里有个至关重要的细节:醒来后第一件事,应该是重新检查等待条件是否真的满足了。

必须配合 while 循环做条件重检

为什么必须重新检查?因为 wait() 存在“虚假唤醒”的可能性——线程可能因为某些与业务逻辑无关的原因(比如操作系统信号干扰)而被唤醒。所以,绝对不能简单地用 if 判断一次就了事,必须用 while 循环把等待条件包起来:

synchronized (sharedObj) {
  while (!conditionMet()) { // 用while,不是if!
    sharedObj.wait();
  }
  // 执行到这里时,conditionMet() 肯定为 true了,可以安全地处理后续逻辑
}

举个典型的例子,在生产者-消费者模型里,消费者线程能行动的前提是“队列非空”。所以,它的等待逻辑必须是 while(!queue.isEmpty()) wait(); —— 即使被唤醒了,也得再瞅一眼队列,确认真的有东西可消费才行。

notify 与 notifyAll 的选择依据

那么,叫醒一个线程,是用 notify() 还是 notifyAll() 呢?这得看场景。

  • notify() 会随机唤醒等待在同一个对象上的一个线程。它适合条件单一、等待方角色明确的场景,比如两个线程严格交替执行。
  • notifyAll() 则更“豪爽”,它会唤醒所有正在等待的线程。这种方式更安全、更通用,尤其是在多个线程等待不同条件(比如有的在等“非空”,有的在等“未满”)的复杂场景下。唤醒后,让每个线程自己用 while 循环去判断是否满足了自己的条件,满足的就继续执行,不满足的继续等待。

对于初学者来说,有个实用的建议:如果不确定该用哪个,那就统一使用 notifyAll()。虽然可能带来一点点性能开销,但它能有效避免因唤醒错对象而导致的死锁风险。当然,如果你能百分之百确定,只有一类线程在等待某个条件,并且你一旦唤醒它,条件肯定满足,那么用 notify() 会更轻量一些。

典型协作模式示例:简单阻塞队列

理论说了这么多,来看一个最经典的“生产一个,消费一个”的协作模式,思路就非常清晰了:

  • 生产者线程的行动路线是:获取锁 → 检查队列是否已满(用while循环判断)→ 如果满了就 wait() → 添加元素 → 调用 notifyAll() → 释放锁。
  • 消费者线程的行动路线是:获取锁 → 检查队列是否为空(同样用while循环判断)→ 如果空了就 wait() → 取出元素 → 调用 notifyAll() → 释放锁。

这个模式里的关键点有两个:一是双方都在修改完共享状态后、离开同步块前调用 notifyAll();二是双方都坚持用 while 循环来重新检查等待条件。把握住这两点,一个基础的线程协作框架就搭建起来了。

本文转载于:https://www.php.cn/faq/2411171.html 如有侵犯,请联系zhengruancom@outlook.com删除。
免责声明:正软商城发布此文仅为传递信息,不代表正软商城认同其观点或证实其描述。

热门关注