IllegalMonitorStateException发生在线程未获取对象锁时调用wait/notify方法,正确做法是在synchronized块中调用或使用Condition等高级并发工具避免该异常。

在Java中,IllegalMonitorStateException 是一种运行时异常,通常发生在使用 wait()、notify() 或 notifyAll() 方法时,当前线程没有持有目标对象的监视器(即未获取该对象的锁)。要正确捕获和处理这个异常,关键在于理解其触发条件并从代码设计上避免。
理解IllegalMonitorStateException的触发原因
这个异常的核心原因是:线程在调用 wait()、notify() 或 notifyAll() 时,没有通过 synchronized 块或方法获得对应对象的锁。JVM要求这些操作必须在同步上下文中执行。
例如以下代码会抛出该异常:
Object lock = new Object(); // 错误:未在synchronized块中调用 lock.wait(); // 抛出IllegalMonitorStateException
正确的做法是确保这些方法在同步块或同步方法内被调用。
立即学习“Java免费学习笔记(深入)”;
如何捕获和预防异常
虽然 IllegalMonitorStateException 是运行时异常,不需要强制捕获,但可以通过编码规范和防御性检查来避免。
- 确保调用
wait()、notify()前,已进入该对象的 synchronized 块或方法 - 使用 try-catch 包裹以进行调试或日志记录(非必需,但有助于排查)
- 优先使用高阶并发工具类替代手动锁管理
示例:安全地使用 wait/notify
synchronized (lock) {
try {
lock.wait(1000); // 正确:已在同步块中
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
}
}
推荐的线程协作替代方案
为减少此类错误,建议使用 java.util.concurrent 包中的工具类,它们封装了底层锁逻辑,更安全易用。
- BlockingQueue:用于线程间安全传递数据,无需手动调用 wait/notify
- CountDownLatch:等待一组操作完成
-
Condition:配合
ReentrantLock使用,提供比 wait/notify 更灵活的等待通知机制
例如使用 Condition:
Lock lock = new ReentrantLock();
Condition cond = lock.newCondition();
lock.lock();
try {
cond.await(); // 安全的等待
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
} finally {
lock.unlock();
}
基本上就这些。只要确保在调用 wait/notify 前持有对象锁,或改用高级并发工具,就能有效避免 IllegalMonitorStateException。问题不复杂,但容易因疏忽引发,多注意同步控制范围即可。










