LockSupport.park()和unpark()提供线程阻塞与唤醒机制,不依赖synchronized,基于许可实现,避免丢失唤醒,支持精确唤醒指定线程,是AQS等同步组件的基础。

在Java中,LockSupport.park() 和 LockSupport.unpark() 是用于线程阻塞与唤醒的底层工具,它们由JUC(java.util.concurrent)包提供,比传统的 wait/notify 更加灵活且不易出错。这两个方法直接作用于线程,不依赖 synchronized 或 monitor 锁,适合实现自定义同步器或高级并发控制。
基本用法说明
LockSupport.park() 会使当前线程进入等待状态,直到其他线程调用 LockSupport.unpark(thread) 将其唤醒,或者当前线程被中断。
LockSupport.unpark(Thread thread) 可以唤醒指定的线程。如果目标线程尚未阻塞,下次调用 park 时会直接返回(许可机制),避免了 lost wake-up 问题。
简单示例:先 unpark 再 park
下面这个例子展示了即使先调用 unpark,后续的 park 也不会阻塞:
立即学习“Java免费学习笔记(深入)”;
Thread t = new Thread(() -> { System.out.println("即将 park"); LockSupport.park(); // 不会阻塞,因为已有许可 System.out.println("已恢复运行");});// 先发放许可LockSupport.unpark(t);t.start();
输出结果为:
即将 park已恢复运行
这是因为 unpark 提供了一个“许可”,park 会消费这个许可并立即返回。
线程间精确唤醒
使用 park/unpark 可以精确地控制哪个线程被唤醒,不像 notify 随机唤醒一个等待线程。
Thread worker = new Thread(() -> { System.out.println("工作线程开始执行任务"); System.out.println("工作线程准备 park"); LockSupport.park(); System.out.println("工作线程被唤醒,继续执行");});worker.start();// 主线程休眠一会儿,确保 worker 先启动try { Thread.sleep(1000); } catch (InterruptedException e) {}System.out.println("主线程调用 unpark 唤醒工作线程");LockSupport.unpark(worker);
输出顺序清晰,能保证 worker 被正确唤醒。
注意事项与常见问题
无对象监视器依赖:park/unpark 不需要 synchronized 块,也不涉及对象的 monitor,因此更轻量。 许可机制:unpark 发放一个许可,最多保留一个;多次 unpark 同一线程效果等同于一次。 无法感知中断原因:park() 被中断后不会抛异常,需手动检查中断状态,可用 LockSupport.parkNanos(long nanos) 或带中断检测的版本配合 Thread.interrupted() 判断。 不能重复 park 同一线程而没有新的 unpark:一旦许可被消费,再次 park 会阻塞,除非再有 unpark。
基本上就这些。LockSupport 的设计简洁高效,是 AQS(AbstractQueuedSynchronizer)等高级同步组件的基础。掌握它有助于理解 Java 并发底层机制,也能在特定场景下写出更精准的线程协作逻辑。
以上就是如何在Java中使用LockSupport.park与unpark的详细内容,更多请关注创想鸟其它相关文章!
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/11252.html
微信扫一扫
支付宝扫一扫