LockSupport是Java中用于线程阻塞与唤醒的核心工具类,位于java.util.concurrent.locks包下,其核心方法为park()和unpark(Thread thread)。park()使当前线程阻塞,直到其他线程调用该线程的unpark方法或线程被中断;unpark()则向指定线程发放许可,允许其继续执行,且可先于park()调用生效。相比wait/notify机制,LockSupport更灵活安全,不依赖synchronized关键字,也不会因顺序问题导致死锁。它基于单一许可机制,重复调用unpark不会累积许可。常用方法包括带阻塞原因的park(Object blocker),便于线程诊断;还有限时阻塞的parkNanos和parkUntil。示例中展示了主线程延迟后唤醒工作线程的过程,输出清晰体现执行流程。使用时需注意:park不抛出InterruptedException,但可通过Thread.interrupted()检测中断状态;应结合条件判断使用,避免盲目阻塞;精确同步需配合状态变量实现。总之,LockSupport是构建高级并发组件如ReentrantLock、CountDownLatch的基础,掌握其机制对深入理解Java并发至关重要。

在Java中,LockSupport 是一个非常底层且实用的线程阻塞工具类,位于 java.util.concurrent.locks 包下。它提供了基本的线程阻塞和唤醒能力,是实现更高级同步器(如 ReentrantLock、CountDownLatch)的基础。相比传统的 wait/notify 和 suspend/resume 机制,LockSupport 更加灵活、安全且不易出错。
LockSupport 核心方法介绍
LockSupport 主要通过两个静态方法来控制线程的阻塞与唤醒:
- LockSupport.park():阻塞当前线程,直到其他线程调用该线程的 unpark 方法,或当前线程被中断。
- LockSupport.unpark(Thread thread):唤醒指定的线程,即使 unpark 先于 park 调用,后续的 park 也不会真正阻塞(相当于“通行证”已发放)。
这种“许可”机制类似于信号量,每个线程最多持有一次许可,多次调用 unpark 也不会叠加许可。
基本使用示例
下面是一个简单的例子,展示如何使用 LockSupport 让一个线程等待,另一个线程唤醒它:
立即学习“Java免费学习笔记(深入)”;
public class LockSupportDemo {
public static void main(String[] args) {
Thread worker = new Thread(() -> {
System.out.println("工作线程开始执行任务...");
System.out.println("工作线程准备阻塞");
LockSupport.park(); // 阻塞自己
System.out.println("工作线程被唤醒,继续执行");
});
worker.start();
try {
Thread.sleep(2000); // 主线程等待2秒
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("主线程唤醒工作线程");
LockSupport.unpark(worker); // 唤醒指定线程
}}
输出结果为:
工作线程开始执行任务...
工作线程准备阻塞
主线程唤醒工作线程
工作线程被唤醒,继续执行
支持带条件的阻塞
除了无参的 park(),LockSupport 还提供了一些变体方法:
- park(Object blocker):阻塞当前线程,并设置阻塞原因(blocker),便于调试和监控。
- parkNanos(long nanos):阻塞指定纳秒数。
- parkUntil(long deadline):阻塞到某个绝对时间点(毫秒时间戳)。
推荐使用 park(Object blocker),因为它能在 JVM 线程转储中显示阻塞原因,有助于排查问题。例如:
LockSupport.park("等待资源释放");在 jstack 输出中会看到类似信息:- parking to wait for
注意事项与技巧
使用 LockSupport 时需注意以下几点:
- unpark 可以在 park 之前调用,不会失效,这是其优势之一。
- park 不会抛出 InterruptedException,但可以通过 Thread.interrupted() 检测中断状态。
- 每个线程只有一个“许可”,重复 unpark 不会累积效果。
- 不能依赖 park/unpark 实现精确的一对一通信逻辑,需结合状态变量使用。
例如,在判断是否需要阻塞时,应先检查条件:
if (!conditionMet()) {
LockSupport.park(this);
}基本上就这些。LockSupport 虽然简单,但它是构建高效并发组件的重要基石,掌握其原理和使用方式对深入理解 Java 并发编程很有帮助。










