LockSupport的源码实现原理(一)

news2025/1/3 8:19:10

目录

底层源码分析

线程状态变化

 许可证机制

中断处理


底层源码分析

public class LockSupport {
    // Unsafe实例
    private static final Unsafe U = Unsafe.getUnsafe();
    
    // Thread对象中parkBlocker字段的偏移量
    private static final long PARKBLOCKER = U.objectFieldOffset(Thread.class, "parkBlocker");
    
    // 私有构造函数,防止实例化
    private LockSupport() {}
}

       由源码中可以看出,LockSupport类有一个Unsafe对象的静态属性,通过Unsafe类的静态方法getUnsafe获取的,而在Unsafe类中就有个静态属性Unsafe,用的是单例模式饿汉式创建。

       也由此可得知,LockSupport的底层其实就是依靠Unsafe类提供的方法。

       接下来看看核心方法。

public class LockSupport {
    
    // park方法实现
    public static void park() {
        // 调用Unsafe的park方法
        U.park(false, 0L);
    }
    
    // 带blocker的park方法
    public static void park(Object blocker) {
        // 获取当前线程
        Thread t = Thread.currentThread();
        // 设置blocker
        setBlocker(t, blocker);
        // 执行park操作
        U.park(false, 0L);
        // 清除blocker
        setBlocker(t, null);
    }
    
    // unpark方法实现
    public static void unpark(Thread thread) {
        if (thread != null)
            U.unpark(thread);
    }
}

          可以看到实际上就是调用了unsafe类的park和unpark方法,至于这个blocker对象也是大有用处的,想要了解的读者可以看看这篇文章LockSupport底层源码分析(二)-CSDN博客,先接着往下看。

         unsafe类的park方法是一个native本地方法,底层是c/c++实现的。

// Hotspot底层实现(C++代码)
void Parker::park(bool isAbsolute, jlong time) {
    // 获取Parker对象
    Parker* p = Parker::fromThread(thread);
    
    // 检查_counter
    if (p->_counter > 0) {
        // 已经有unpark了,直接返回
        p->_counter = 0;
        return;
    }
    
    // 设置等待状态
    ThreadState old = thread->set_state(WAITING);
    
    // 等待条件变量
    while (p->_counter == 0) {
        if (time > 0) {
            // 带超时的等待
            p->_cond.wait(time);
        } else {
            // 无限等待
            p->_cond.wait();
        }
    }
    
    // 恢复线程状态
    thread->set_state(old);
    
    // 重置计数器
    p->_counter = 0;
}

         在上可以看到c++源码实现的一个情况,获取到thread关联的parker对象,检查他的一个counter属性是不是大于0,大于0则已经有unpark了,直接返回。不然就设置线程状态,再调用wait方法进行等待,如果被unpark了或者是到时间了,再恢复线程状态,重置计时器。

         这也说明了为什么park方法不会被interrupt打断,因为他也根本没做任何的响应操作。

         再看看下面的代码实现。

线程状态变化

// 线程状态转换示意
public class ThreadStateDemo {
    public void demonstrateThreadStates() {
        Thread t = new Thread(() -> {
            // 运行状态 RUNNABLE
            
            LockSupport.park();
            // 等待状态 WAITING
            
            // 被unpark后恢复到 RUNNABLE
        });
        
        t.start();
        // 某个时刻调用unpark
        LockSupport.unpark(t);
    }
}

 许可证机制

public class PermitExample {
    public void demonstratePermit() {
        Thread t = new Thread(() -> {
            // 1. 先park
            System.out.println("准备park");
            LockSupport.park();
            System.out.println("park结束");
            
            // 2. 再次park
            System.out.println("再次park");
            LockSupport.park();
            System.out.println("第二次park结束");
        });
        
        t.start();
        
        // 提前调用unpark
        LockSupport.unpark(t);
        // 线程启动后会直接通过第一个park
        
        Thread.sleep(1000);
        // 再次unpark
        LockSupport.unpark(t);
        // 线程通过第二个park
    }
}

中断处理

public class InterruptExample {
    public void demonstrateInterrupt() {
        Thread t = new Thread(() -> {
            try {
                // park不响应中断,但会记录中断状态
                LockSupport.park();
                
                // 检查中断状态
                if (Thread.interrupted()) {
                    System.out.println("被中断");
                    return;
                }
                
                // 继续执行
                doWork();
            } catch (Exception e) {
                e.printStackTrace();
            }
        });
        
        t.start();
        
        // 中断线程
        t.interrupt();
    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2268994.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

CannotRetrieveUpdates alert in disconnected OCP 4 cluster解决

环境: Red Hat OpenShift Container Platform (RHOCP) 4 问题: Cluster Version Operator 不断发送警报,表示在受限网络/断开连接的 OCP 4 集群中无法接收更新。 在隔离的 OpenShift 4 集群中看到 CannotRetrieveUpdates 警报: …

智能家居体验大变革 博联 AI 方案让智能不再繁琐

1. 全球AI技术发展背景及智能家居市场趋势 人工智能(AI)技术的飞速发展正在推动全球各行业的数字化转型。国际电信联盟与德勤联合发布《人工智能向善影响》报告指出,全球94%的商界领袖认为,人工智能技术对于其企业在未来5年内的发…

logback日志框架源码分析

目录 (一)入口:slf4j选择日志框架 (二)日志框架初始化 (1)logback的3种配置方式 a、BasicConfigurator默认配置 b、SPI方式配置的Configurator实现类 c、通过配置文件初始化 (2)xml配置文件初始化 (三)Logger的创建 (四)打印日志 本文源码基于:logback版…

VScode 只能运行c,运行不了c++的解决问题

原文链接:Vscode只能运行c,运行不了c的解决方法 VScode 只能运行c,运行不了c,怎么回事呢,解决问题: 在tasks.json中加上“"-lstdc"”, 这样之后 要重启VScode,点击链接…

HTML——45.单元格合并

<!DOCTYPE html> <html><head><meta charset"UTF-8"><title>表格</title></head><body><!--合并单元格&#xff1a;1.在代码中找到要合并的单元格2.在要合并的所有单元格中&#xff0c;保留要合并的第一个单元格…

记录第一次跑YOLOV8做目标检测

今天是24年的最后一天&#xff0c;终于要向新世界开始破门了&#xff0c;开始深度学习&#xff0c;YOLO来敲门~ 最近做了一些皮肤检测的功能&#xff0c;在传统的处理中经历了反复挣扎&#xff0c;终于要上YOLO了。听过、看过&#xff0c;不如上手体会过~ 1、YOLO是什么&#x…

如何配置Java应用程序的远程调试

配置Java应用程序的远程调试涉及几个步骤&#xff0c;包括在启动Java应用程序时设置特定的JVM参数&#xff0c;以及在IDE&#xff08;如Eclipse、IntelliJ IDEA等&#xff09;中配置远程调试连接。以下是详细的步骤&#xff1a; 启动Java应用程序时启用远程调试 为了能够远程…

2025考研江南大学复试科目控制综合(初试807自动控制原理)

​ 2025年全国硕士研究生招生考试江南大学考点 一年年的考研如期而至&#xff0c;我也变成了研二了&#xff0c;作为2次考研经历的学长&#xff0c;总是情不自禁地回想起自己的考研经历&#xff0c;我也会经常从那段经历中汲取力量。我能理解大多数考生考完后的的迷茫无助&…

时间序列预测算法---LSTM

文章目录 一、前言1.1、深度学习时间序列一般是几维数据&#xff1f;每个维度的名字是什么&#xff1f;通常代表什么含义&#xff1f;1.2、为什么机器学习/深度学习算法无法处理时间序列数据?1.3、RNN(循环神经网络)处理时间序列数据的思路&#xff1f;1.4、RNN存在哪些问题?…

[羊城杯 2024]不一样的数据库_2

题目描述&#xff1a; 压缩包6 (1).zip需要解压密码&#xff1a; 尝试用ARCHPR工具爆破一下&#xff1a; &#xff08;字典可自行在github上查找&#xff09; 解压密码为&#xff1a;753951 解压得到13.png和Kee.kdbx文件&#xff1a; 二维码图片看上去只缺了正常的三个角&…

图像处理-Ch7-小波函数

个人博客&#xff01;无广告观看&#xff0c;因为这节内容太多了&#xff0c;有点放不下&#xff0c;分了三节 文章目录 多分辨率展开(Multi-resolution Expansions)序列展开(Series Expansions)尺度函数(Scaling Function)例&#xff1a;哈尔尺度函数(Haar scaling func)多分…

CPT203 Software Engineering 软件工程 Pt.2 敏捷方法和需求工程(中英双语)

文章目录 3. Aglie methods&#xff08;敏捷方法&#xff09;3.1 Aglie methods&#xff08;敏捷方法&#xff09;3.1.1 特点3.1.2 优点3.1.3 缺点3.1.4 原则3.1.5 计划驱动与敏捷方法的对比 3.2 Scrum3.2.1 Scrum roles3.2.2 Scrum Activities and Artifacts3.2.2.1 Product B…

【人工智能机器学习基础篇】——深入详解深度学习之神经网络基础:理解前馈神经网络与反向传播算法

深入详解深度学习之神经网络基础&#xff1a;理解前馈神经网络与反向传播算法 深度学习作为人工智能&#xff08;AI&#xff09;的核心技术&#xff0c;已经在语音识别、图像处理、自然语言处理等诸多领域取得了显著的成果。而在深度学习的众多模型中&#xff0c;**前馈神经网络…

【循环神经网络】RNN介绍

在人工神经网络中&#xff0c;”浅层网络”是指具有一个输入层、一个输出层和最多一个没有循环连接的隐藏层的网络。随着层数的增加&#xff0c;网络的复杂性也在增加。更多的层或循环连接通常会增加网络的深度&#xff0c;并使其能够提供不同级别的数据表示和特征提取&#xf…

【论文投稿】Python 网络爬虫:探秘网页数据抓取的奇妙世界

【IEEE出版|广东工业大学主办】第五届神经网络、信息与通信工程国际学术会议&#xff08;NNICE 2025&#xff09;_艾思科蓝_学术一站式服务平台 目录 前言 一、Python—— 网络爬虫的绝佳拍档 二、网络爬虫基础&#xff1a;揭开神秘面纱 &#xff08;一&#xff09;工作原…

基于深度学习的视觉检测小项目(二) 环境和框架搭建

一、环境和框架要求 SAM的环境要求&#xff1a; Python>3.7 PyTorch>1.7 torchvision>0.8 YOLO V8的环境要求&#xff1a;YOLO集成在ultralytics库中&#xff0c;ultralytics库的环境要求&#xff1a; Python>3.7 PyTorch>1.10.0 1、确定pytorch版本…

SQLiteDataBase数据库

XML界面设计 <?xml version"1.0" encoding"utf-8"?> <LinearLayout xmlns:android"http://schemas.android.com/apk/res/android"xmlns:tools"http://schemas.android.com/tools"android:layout_width"match_paren…

cesium小知识: 处理动画的5种方式

在 Cesium 中处理动画可以通过多种方式实现,具体取决于你想要创建的动画类型。Cesium 提供了丰富的API来支持不同种类的动画,包括但不限于物体的移动、旋转、缩放、属性变化等。以下是几种常见的动画处理方法: 1. 使用 Entity 和 SampledProperty 对于动态数据或随时间变化…

【鸿蒙NEXT】鸿蒙里面类似iOS的Keychain——关键资产(@ohos.security.asset)实现设备唯一标识

前言 在iOS开发中Keychain 是一个非常安全的存储系统&#xff0c;用于保存敏感信息&#xff0c;如密码、证书、密钥等。与 NSUserDefaults 或文件系统不同&#xff0c;Keychain 提供了更高的安全性&#xff0c;因为它对数据进行了加密&#xff0c;并且只有经过授权的应用程序才…

visual studio连接sql server数据库

目录 1、为什么要建立连接2、在sql server中建立数据库3、visual studio连接sql server数据库4、学生信息管理系统页面布局5、添加事件逻辑 5.1 页面跳转5.2 读取学生信息5.3 查询学生信息5.4 修改学生信息5.5 删除学生信息5.6 添加学生信息 bilibili演示视频 github源码 1、…