LockSupport通过park和unpark实现线程阻塞与唤醒,支持中断响应且可传递诊断信息,常用于构建高级同步器如AQS。
在Java中,LockSupport 是一个非常底层的线程阻塞与唤醒工具类,位于 java.util.concurrent.locks 包下。它不像 synchronized 或 ReentrantLock 那样提供显式的锁机制,而是直接对线程进行“停车”和“发车”操作,常用于构建更高级的同步器,比如 AQS(AbstractQueuedSynchronizer)。下面介绍如何使用 LockSupport 来控制线程的挂起与唤醒。
LockSupport 的核心方法是两个静态方法:
与 wait/notify 不同,unpark 可以在 park 之前调用,相当于“预发通行证”。如果线程已经持有许可,park 调用会立刻返回,不会阻塞。
示例代码:
Thread t = new Thread(() -> { System.out.println("线程即将 park"); LockSupport.park(); System.out.println("线程被唤醒"); }); t.start(); // 主线程休眠1秒,确保子线程先执行 try { Thread.sleep(1000); } catch (InterruptedException e) {} System.out.println("主线程调用 unpark"); LockSupport.unpark(t);
输出结果:
线程即将 park 主线程调用 unpark 线程被唤醒
调用 park() 的线程如果被中断,不会抛出 InterruptedException,但会立即返回。可以通过 Thread.interrupted() 检查中断状态。
例如:
Thread t = new Thread(() -> {
System.out.println("进入 park");
LockSupport.park();
if (Thread.interrupted()) {
System.out.println("线程被中断了");
}
System.out.println("继续执行");
});
注意:interrupted() 是清除中断状态的方法,慎用。
你可以传入一个“阻塞原因”对象给 park 方法,便于调试线程状态:
LockSupport.park("等待资源释放");
这个信息可以在 JVM 线程转储(thread dump)中看到,有助于排查问题。
虽然 LockSupport 很灵活,但也容易误用:
常见错误:忘记 unpark 或线程已退出导致无法唤醒。
基本上就这些。LockSupport 是 Java 并发包的基石之一,理解它有助于深入掌握并发原理。用起来简单,但要小心控制线程生命周期和调用顺序。