LockSupport无故放开锁的问题

本文探讨了在Java中使用LockSupport.park()方法时出现的异常行为,即使没有调用unpark()方法,控制台仍打印出over字段。通过对JDK源码的深入分析,发现park()方法可能存在无条件返回的情况。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

本文为笔者在学习过程中遇到的未解决的问题,请各位大牛帮忙在评论区或联系笔者解答

代码如下,问题见注释

	/**
     * 全程没有LockSupport.unpark()代码,但是控制台打印出了over字段
     * LockSupport.park()在线程中并且之前有线程被中断时无法加锁
     *
     * 环境:jdk1.8,windows8
     */
    private void testPermitBug() {
        Thread thread1 = new Thread(()->{
            try {
                Thread.sleep(5000);
            } catch (InterruptedException e) {
                System.out.println("睡眠被中断");
            }
        });
        thread1.start();
        waitAMoment();
        thread1.interrupt();

        Thread thread2 = new Thread(()->{
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            LockSupport.park();
            System.out.println("over");
        });
        thread2.start();
        waitAMoment();
        printThreadState(thread2);
    }

运行结果

在这里插入图片描述

个人见解

在jdk源码LockSupport.java的第160行发现了这一句

<li>The call spuriously (that is, for no reason) returns.

这是park()方法的注释,说这个方法可能会无条件结束。。。不是吧

Github博客

### Java LockSupport 使用方法及原理 #### 一、LockSupport 类概述 `LockSupport` 是 JDK 提供的一个工具类,主要用于线程的阻塞与唤醒操作。它通过 `park()` 和 `unpark()` 方法实现了对单个线程的精确控制[^2]。 #### 二、核心方法解析 1. **`park()` 方法** - 调用此方法会使当前线程进入阻塞状态,直到发生以下情况之一才会被唤醒: - 另一线程调用了该线程对应的 `unpark()` 方法; - 当前线程中断; - 设置了一个超时时间(如果适用)。 - 阻塞过程无需依赖任何对象,因此更加灵活和高效[^3]。 2. **`unpark(Thread thread)` 方法** - 此方法用于唤醒指定线程上的 `park()` 阻塞。 - 如果目标线程尚未调用 `park()` 或者已经退出,则此方法不会产生任何效果。 - 值得注意的是,`unpark()` 可以在 `park()` 之前调用,这种情况下会为下一次的 `park()` 提前设置许可标志位[^4]。 #### 三、工作原理 `LockSupport` 的底层机制基于操作系统级别的线程调度接口实现。具体而言,它的功能由 `Unsafe` 类中的 native 方法支持,这些方法允许 JVM 对线程进行更精细的操作[^1]。以下是其实现的关键点: - **Native 层面的支持** 在 Linux 平台上,`LockSupport` 实际上利用了 POSIX 线程库 (`pthread`) 中的相关 API 来完成线程挂起与恢复的工作[^5]。 - **Permit 许可机制** 每个线程都拥有一个关联的 permit(许可证)。当调用 `unpark(thread)` 后,对应线程的 permit 将会被置为可用;而一旦线程执行到 `park()`,就会尝试消费这个 permit。如果没有可用的 permit,则线程将进入等待队列直至获得许可。 #### 四、使用示例 下面是一个简单的例子展示如何使用 `LockSupport` 进行线程间的协作: ```java public class LockSupportExample { public static void main(String[] args) throws InterruptedException { Thread t = new Thread(() -> { System.out.println("Thread is about to park."); LockSupport.park(); // 主动让线程暂停运行 System.out.println("Thread unparked and resumed!"); }); t.start(); // 给主线程一点时间启动子线程 Thread.sleep(1000); System.out.println("Unparking the parked thread..."); LockSupport.unpark(t); // 解除子线程的阻塞状态 } } ``` #### 五、与其他同步方式对比 | 特性 | `LockSupport` | `synchronized/wait-notify` | |---------------------|----------------------------------|-----------------------------------| | 是否需要对象 | 不需要 | 需要 | | 控制粒度 | 单线程层面 | 多线程层面 | | 唤醒策略 | 精确唤醒 | 随机或全部唤醒 | | 调用顺序灵活性 | 支持提前调用 `unpark` | 必须先有 `wait` 才能有效 | 上述表格总结了两者的主要差异。 --- ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值