InterruptedException 是Java中表示线程被中断的受检异常,需响应而非忽略:要么退出、要么恢复中断状态(Thread.currentThread().interrupt())、要么向上抛出,否则中断信号丢失。

InterruptedException 是什么,为什么不能简单 catch 了事
InterruptedException 是 Java 中一个受检异常(checked exception),只会在线程被其他线程调用 Thread.interrupt() 后,且当前线程正处于阻塞状态(如 sleep()、wait()、join()、LockSupport.park() 等)时抛出。它不是“错误”,而是协作式中断机制的信号——意味着“有人希望你停下来”。直接 catch 后空着不处理,或者只打日志,等于丢弃中断请求,后续调用 Thread.currentThread().isInterrupted() 仍可能返回 false,导致中断丢失。
正确响应中断的三种典型做法
关键原则:要么退出当前逻辑,要么重新设置中断状态,让上层代码能感知到中断意图。
- 在可中断的循环中检查并退出:
while (!Thread.currentThread().isInterrupted()) { try { doWork(); Thread.sleep(1000); } catch (InterruptedException e) { // 清理后主动退出循环 Thread.currentThread().interrupt(); // 恢复中断状态(可选,若不再抛出) break; } } - 在方法签名中向上抛出:
public void doSomething() throws InterruptedException { Thread.sleep(500); // 不捕获,让调用方决定如何响应 }适用于工具方法或框架回调(如Runnable实现类中不建议抛出,因接口不声明该异常) - 捕获后立即恢复中断状态并返回或抛出自定义异常:
try { queue.take(); // 可能阻塞 } catch (InterruptedException e) { Thread.currentThread().interrupt(); // 必须!否则中断丢失 throw new TaskCancelledException("Task interrupted", e); }
常见踩坑点:interrupt() 和 isInterrupted() 的区别
Thread.interrupt() 是向目标线程发出中断请求;而 Thread.isInterrupted() 只是查询当前中断状态(不清除标志位)。但 Thread.currentThread().interrupted()(注意是静态方法)不仅查询,还会清除中断状态——这是易错点。
- 错误写法:
if (Thread.interrupted()) { // 清除了状态!后面再调用就返回 false return; } // … 后续又想检查?已经晚了 if (Thread.interrupted()) { // 总是 false cleanup(); } - 正确写法(保持状态可重入检查):
Thread t = Thread.currentThread(); if (t.isInterrupted()) { cleanup(); return; } - 所有阻塞方法(如
sleep())在抛出InterruptedException前,都会自动清除中断状态。所以捕获后若不手动interrupt(),状态就没了。
ExecutorService 中的中断容易被忽略的细节
提交到线程池的任务,如果没响应中断,shutdownNow() 可能无法真正终止它们。
立即学习“Java免费学习笔记(深入)”;
-
Future.cancel(true)本质就是对执行该任务的线程调用interrupt(),但仅当任务处于阻塞态时才生效;如果任务正在跑纯计算逻辑(无阻塞调用),中断不会起作用。 - 不要依赖
executor.shutdownNow()立即停止所有任务——它只是发中断+尝试停止正在运行的 Worker 线程,具体是否停得下来,取决于每个任务内部是否响应中断。 - 使用
CompletableFuture时,cancel(true)对底层线程无效,它只影响 future 自身状态;需自行在异步逻辑中检查中断。
Thread.currentThread().interrupt() 这一行,往往就是整个中断逻辑失效的根源。









