LockSupport是Java中基于许可机制的线程阻塞与唤醒工具,核心方法park()和unpark(Thread)通过Unsafe实现,无需锁即可精准控制线程,具备顺序无关性与高灵活性,广泛用于AQS等并发框架。

在Java中,LockSupport 是一个非常基础且高效的线程阻塞与唤醒工具类,位于 java.util.concurrent.locks 包下。它不像传统的 synchronized 或 Object.wait/notify 那样依赖对象监视器,也不像 Condition 需要配合 ReentrantLock 使用,而是直接对线程进行操作,更加灵活。
LockSupport 的核心方法
实现线程挂起与唤醒主要依赖两个静态方法:
- LockSupport.park():挂起当前线程,直到收到唤醒信号或其他中断事件。
- LockSupport.unpark(Thread thread):唤醒指定的线程。
这两个方法底层基于 Unsafe 类实现,使用了“许可”(permit)机制:每个线程最多拥有一个许可。调用 unpark 会为线程发放一个许可,而 park 会尝试消耗这个许可。如果许可已存在,park 不会阻塞;如果没有,则线程进入阻塞状态。
基本使用示例:线程挂起与唤醒
下面是一个简单的例子,展示如何使用 LockSupport 控制线程执行顺序:
立即学习“Java免费学习笔记(深入)”;
public class LockSupportDemo {
public static void main(String[] args) {
Thread worker = new Thread(() -> {
System.out.println("工作线程开始执行...");
// 挂起自己
LockSupport.park();
System.out.println("工作线程被唤醒,继续执行任务。");
}, "WorkerThread");
worker.start();
try {
Thread.sleep(2000); // 主线程等待2秒
} catch (InterruptedException e) {
e.printStackTrace();
}
// 唤醒指定线程
System.out.println("主线程准备唤醒工作线程。");
LockSupport.unpark(worker);
}
}
输出结果:
工作线程开始执行...主线程准备唤醒工作线程。
工作线程被唤醒,继续执行任务。
注意:即使 unpark 在 park 之前调用,也不会失效,因为许可已经存在,后续的 park 调用会立即返回,不会阻塞。
park/unpark 的优势对比 wait/notify
相比传统的 wait/notify,LockSupport 有以下明显优点:
- 无需获取锁:调用 park 不需要像 wait 那样先持有 synchronized 锁。
- 线程精准控制:unpark 可以指定具体线程,而 notify 是随机唤醒一个等待线程。
- 顺序无关性:先 unpark 再 park 不会死锁,许可机制保证了正确性。
- 更简洁直观:API 简单,逻辑清晰,适合构建高级同步组件。
常见注意事项
使用 LockSupport 时需注意以下几点:
- 多次调用 unpark 只会保留一个许可,多余的无效。
- park 方法响应中断,但不会抛出 InterruptedException,需通过 Thread.interrupted() 判断中断状态。
- 避免在循环中无条件 park,应结合状态判断防止误唤醒或遗漏唤醒。
基本上就这些。LockSupport 虽然简单,却是 AQS(AbstractQueuedSynchronizer)等并发框架的基石,理解其原理有助于深入掌握 Java 并发编程。不复杂但容易忽略细节。










