Java中调用wait()必须在synchronized块内,用while循环检查条件,notify/notifyAll也需在同步块中与条件更新原子执行,优先使用notifyAll()避免信号丢失。
在Java中调用 wait() 方法不是简单地“让线程等一下”,它必须严格配合 synchronized 块和 notify()/notifyAll() 使用,否则极易导致死锁、IllegalMonitorStateException 或虚假唤醒等问题。
wait() 是 Object 类的实例方法,它的语义是“释放当前对象锁并挂起线程”。JVM 要求调用线程必须已持有该对象的监视器锁(即已进入 synchronized 代码块或方法),否则会立即抛出 IllegalMonitorStateException。
wait()、notify()、notifyAll() 必须作用于同一个锁对象,且该对象不能为 null线程被唤醒后,不能假设条件一定满足——可能因虚假唤醒(spurious wakeup)、通知被其他线程消费,或条件在唤醒后又被修改。只用 if 判断会导致线程继续执行错误逻辑。
if (!condition) { obj.wait(); }
while (!condition) { obj.wait(); }
notify() 只随机唤醒一个等待线程,适用于“单生产者-单消费者”且等待条件互斥的场景;多数情况下应优先使用 notifyAll()。
notify() 的风险:若唤醒的线程发现条件仍不满足,它会再次 wait;而真正需要被唤醒的线程可能一直沉睡,造成信号丢失或死锁notifyAll()
不仅 wait() 需要锁,notify() 和 notifyAll() 同样要求调用线程持有对应对象的锁。这是为了保证“修改条件 + 发送通知”这一过程的原子性,防止通知早于条件更新被接收。
sy
nchronized(obj) { condition = true; obj.notifyAll(); }