【Java开发日记】我们来说说 LockSupport 的 park 和 unpark

目录

一、LockSupport

1.1、LockSupport函数列表

1.2、基本使用

先 park 再 unpark

先 unpark 再 park

1.3、特点

与 Object 的 wait & notify 相比

二、LockSupport park & unpark原理

2.1、情况一,先调用park,再调用unpark

park 操作

unpark 操作

2.2、情况二,先调用unpark,再调用park

三、LockSupport Java源码解析

3.1 变量说明

变量是如何获取其实例对象的?

3.2 构造方法

3.3 两个特殊的方法

3.4 常用方法

1、unpark(Thread thread)方法

2、park(Object blocker)方法 和 park()方法

park(Object blocker)函数中要调用两次setBlocker函数

3、parkNanos(Object blocker, long nanos)方法 和 parkNanos(long nanos)方法

4、parkUntil(Object blocker, long deadline)方法 和 parkUntil(long deadline)方法

总结:


一、LockSupport

LockSupport是JDK中比较底层的类,用来创建锁和其他同步工具类的基本线程阻塞原语。
Java锁和同步器框架的核心AQS:AbstractQueuedSynchronizer,就是通过调用LockSupport.park()LockSupport.unpark()实现线程的阻塞和唤醒的。LockSupport很类似于二元信号量(只有1个许可证可供使用),如果这个许可还没有被占用,当前线程获取许可并继续执行;如果许可已经被占用,当前线程阻塞,等待获取许可。
LockSupport中的park() 和 unpark() 的作用分别是阻塞线程和解除阻塞线程,而且park()unpark()不会遇到“Thread.suspend 和 Thread.resume所可能引发的死锁”问题。因为park() 和 unpark()有许可的存在;调用 park() 的线程和另一个试图将其 unpark() 的线程之间的竞争将保持活性。 

1.1、LockSupport函数列表

public class LockSupport {// 返回提供给最近一次尚未解除阻塞的 park 方法调用的 blocker 对象,如果该调用不受阻塞,则返回 null。static Object getBlocker(Thread t);// 为了线程调度,禁用当前线程,除非许可可用。static void park();// 为了线程调度,在许可可用之前禁用当前线程。static void park(Object blocker);// 为了线程调度禁用当前线程,最多等待指定的等待时间,除非许可可用。static void parkNanos(long nanos);// 为了线程调度,在许可可用前禁用当前线程,并最多等待指定的等待时间。static void parkNanos(Object blocker, long nanos);// 为了线程调度,在指定的时限前禁用当前线程,除非许可可用。static void parkUntil(long deadline);// 为了线程调度,在指定的时限前禁用当前线程,除非许可可用。static void parkUntil(Object blocker, long deadline);// 如果给定线程的许可尚不可用,则使其可用。static void unpark(Thread thread);
}

说明:LockSupport是通过调用Unsafe函数中的接口实现阻塞和解除阻塞的。 

1.2、基本使用

// 暂停当前线程
LockSupport.park();
// 恢复某个线程的运行
LockSupport.unpark(暂停线程对象)

先 park 再 unpark
Thread t1 = new Thread(() -> {log.debug("start...");sleep(1);log.debug("park...");LockSupport.park();log.debug("resume...");
},"t1");
t1.start();
sleep(2);
log.debug("unpark...");
LockSupport.unpark(t1);

输出:

18:42:52.585 c.TestParkUnpark [t1] - start...
18:42:53.589 c.TestParkUnpark [t1] - park...
18:42:54.583 c.TestParkUnpark [main] - unpark...
18:42:54.583 c.TestParkUnpark [t1] - resume...

先 unpark 再 park
Thread t1 = new Thread(() -> {log.debug("start...");sleep(2);log.debug("park...");LockSupport.park();log.debug("resume...");
}, "t1");
t1.start();
sleep(1);
log.debug("unpark...");
LockSupport.unpark(t1);

输出:

18:43:50.765 c.TestParkUnpark [t1] - start...
18:43:51.764 c.TestParkUnpark [main] - unpark...
18:43:52.769 c.TestParkUnpark [t1] - park...
18:43:52.769 c.TestParkUnpark [t1] - resume...

1.3、特点

在调用对象的Wait之前当前线程必须先获得该对象的监视器(Synchronized),被唤醒之后需要重新获取到监视器才能继续执行。而LockSupport并不需要获取对象的监视器。 

与 Object 的 wait & notify 相比
  • 1、wait,notify 和 notifyAll 必须配合 Object Monitor 一起使用,而 park,unpark 不必。
  • 2、park & unpark 是以线程为单位来【阻塞】和【唤醒】线程,而 notify 只能随机唤醒一个等待线程,notifyAll是唤醒所有等待线程,但不那么【精确】。
  • 3、park & unpark 可以先 unpark,而 wait & notify 不能先 notify。

因为它们本身的实现机制不一样,所以它们之间没有交集,也就是说LockSupport阻塞的线程,notify/notifyAll没法唤醒。
虽然两者用法不同,但是有一点, LockSupport 的park和Object的wait一样也能响应中断。

public class LockSupportTest {public static void main(String[] args) throws InterruptedException {Thread t = new Thread(() -> {LockSupport.park();System.out.println("thread:"+Thread.currentThread().getName()+"awake");},"t1");t.start();Thread.sleep(2000);//中断t.interrupt();}
}

二、LockSupport park & unpark原理

每个线程都会关联一个 Parker 对象,每个 Parker 对象都各自维护了三个角色:_counter(计数器)、 _mutex(互斥量)、_cond(条件变量)。 

2.1、情况一,先调用park,再调用unpark

park 操作

  1. 当前线程调用 Unsafe.park() 方法
  2. 检查 _counter ,本情况为 0,这时,获得 _mutex 互斥锁
  3. 线程进入 _cond 条件变量阻塞
  4. 设置 _counter = 0 

    unpark 操作
  5. 调用 Unsafe.unpark(Thread_0) 方法,设置 _counter 为 1

  6. 唤醒 _cond 条件变量中的 Thread_0
  7. Thread_0 恢复运行
  8. 设置 _counter 为 0 

    2.2、情况二,先调用unpark,再调用park
  9. 调用 Unsafe.unpark(Thread_0) 方法,设置 _counter 为 1

  10. 当前线程调用 Unsafe.park() 方法
  11. 检查 _counter ,本情况为 1,这时线程无需阻塞,继续运行
  12. 设置 _counter 为 0 

    三、LockSupport Java源码解析

    3.1 变量说明

    public class LockSupport {// Hotspot implementation via intrinsics API//unsafe常量,设置为使用Unsafe.compareAndSwapInt进行更新//UNSAFE字段表示sun.misc.Unsafe类,一般程序中不允许直接调用private static final sun.misc.Unsafe UNSAFE;//表示parkBlocker在内存地址的偏移量private static final long parkBlockerOffset;//表示threadLocalRandomSeed在内存地址的偏移量,此变量的作用暂时还不了解private static final long SEED;//表示threadLocalRandomProbe在内存地址的偏移量,此变量的作用暂时还不了解private static final long PROBE;//表示threadLocalRandomSecondarySeed在内存地址的偏移量// 作用是 可以通过nextSecondarySeed()方法来获取随机数private static final long SECONDARY;
    }

    变量是如何获取其实例对象的?
    public class LockSupport {static {try {//实例化unsafe对象UNSAFE = sun.misc.Unsafe.getUnsafe();Class<?> tk = Thread.class;//利用unsafe对象来获取parkBlocker在内存地址的偏移量parkBlockerOffset = UNSAFE.objectFieldOffset(tk.getDeclaredField("parkBlocker"));//利用unsafe对象来获取threadLocalRandomSeed在内存地址的偏移量SEED = UNSAFE.objectFieldOffset(tk.getDeclaredField("threadLocalRandomSeed"));//利用unsafe对象来获取threadLocalRandomProbe在内存地址的偏移量  PROBE = UNSAFE.objectFieldOffset(tk.getDeclaredField("threadLocalRandomProbe"));//利用unsafe对象来获取threadLocalRandomSecondarySeed在内存地址的偏移量  SECONDARY = UNSAFE.objectFieldOffset(tk.getDeclaredField("threadLocalRandomSecondarySeed"));} catch (Exception ex) { throw new Error(ex); }}
    }

    由上面代码可知这些变量是通过static代码块在类加载的时候就通过unsafe对象获取其在内存地址的偏移量了。 

    3.2 构造方法

    public class LockSupport {//LockSupport只有一个私有构造函数,无法被实例化。private LockSupport() {} // Cannot be instantiated.
    }

    3.3 两个特殊的方法

    public class LockSupport {//设置线程t的parkBlocker字段的值为argprivate static void setBlocker(Thread t, Object arg) {// Even though volatile, hotspot doesn't need a write barrier here.//尽管hotspot易变,但在这里并不需要写屏障。UNSAFE.putObject(t, parkBlockerOffset, arg);}//获取当前线程的Blocker值public static Object getBlocker(Thread t) {//若当前线程为空就抛出异常if (t == null)throw new NullPointerException();//利用unsafe对象获取当前线程的Blocker值 return UNSAFE.getObjectVolatile(t, parkBlockerOffset);}
    }

    1、unpark(Thread thread)方法
    public class LockSupport {//释放该线程的阻塞状态,即类似释放锁,只不过这里是将许可设置为1public static void unpark(Thread thread) {//判断线程是否为空if (thread != null)//释放该线程许可UNSAFE.unpark(thread);}
    }

    2、park(Object blocker)方法 和 park()方法
    public class LockSupport {//阻塞当前线程,并且将当前线程的parkBlocker字段设置为blockerpublic static void park(Object blocker) {//获取当前线程Thread t = Thread.currentThread();//将当前线程的parkBlocker字段设置为blockersetBlocker(t, blocker);//阻塞当前线程,第一个参数表示isAbsolute,是否为绝对时间,第二个参数就是代表时间UNSAFE.park(false, 0L);//重新可运行后再此设置BlockersetBlocker(t, null);}//无限阻塞线程,直到有其他线程调用unpark方法public static void park() {UNSAFE.park(false, 0L);}   
    }

    说明:

所以,park(Object)型函数里必须要调用setBlocker函数两次。 

3、parkNanos(Object blocker, long nanos)方法 和 parkNanos(long nanos)方法
public class LockSupport {//阻塞当前线程nanos秒public static void parkNanos(Object blocker, long nanos) {//先判断nanos是否大于0,小于等于0都代表无限等待if (nanos > 0) {//获取当前线程Thread t = Thread.currentThread();//将当前线程的parkBlocker字段设置为blockersetBlocker(t, blocker);//阻塞当前线程现对时间的nanos秒UNSAFE.park(false, nanos);//将当前线程的parkBlocker字段设置为nullsetBlocker(t, null);}}   //阻塞当前线程nanos秒,现对时间public static void parkNanos(long nanos) {if (nanos > 0)UNSAFE.park(false, nanos);}   
}

4、parkUntil(Object blocker, long deadline)方法 和 parkUntil(long deadline)方法
public class LockSupport {//将当前线程阻塞绝对时间的deadline秒,并且将当前线程的parkBlockerOffset设置为blockerpublic static void parkUntil(Object blocker, long deadline) {//获取当前线程Thread t = Thread.currentThread();//设置当前线程parkBlocker字段设置为blockersetBlocker(t, blocker);//阻塞当前线程绝对时间的deadline秒UNSAFE.park(true, deadline);//当前线程parkBlocker字段设置为nullsetBlocker(t, null);}//将当前线程阻塞绝对时间的deadline秒public static void parkUntil(long deadline) {UNSAFE.park(true, deadline);}   
}

总结:

LockSupport 和 CAS 是Java并发包中很多并发工具控制机制的基础,它们底层其实都是依赖Unsafe实现。很多锁的类都是基于LockSupport的park和unpark来实现的,所以了解LockSupport类是非常重要的。

如果小假的内容对你有帮助,请点赞评论收藏。创作不易,大家的支持就是我坚持下去的动力!

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

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

相关文章

AGI|从“实验室”到“生产线”:企业级AI Agent 如何突围

在数字化转型的深水区&#xff0c;企业级 AI Agent 正从技术概念走向产业实践&#xff0c;成为驱动生产力变革的核心引擎。目录 一、风口已至&#xff1a;AI Agent 的崛起逻辑与市场刚需 二、企业级AI Agent&#xff1a;核心能力与独特价值定位 三、AI Agent 的未来目标 一、…

AtCoder Beginner Contest 417

文章目录A A SubstringB Search and DeleteC Distance IndicatorsD Takahashis ExpectationE A Path in A DictionaryF Random GatheringG Binary CatAtCoder Beginner Contest 417A A Substring You are given an N-character string S consisting of lowercase English lett…

C++23 Concepts:用类型约束重构泛型编程的终极方案

一、开篇:模板元编程的"类型检查困局" 某金融量化团队曾遇到诡异bug: template<typename T> void process(T data) {static_assert(std::is_arithmetic<T>::value, "需要数值类型");// 业务逻辑... } 当调用process("hello")时…

【RK3568 看门狗驱动开发详解】

RK3568 看门狗驱动开发详解一、Linux 看门狗子系统架构​二、设备树配置​三、 看门狗驱动实现四、验证看门狗定时器&#xff08;Watchdog Timer&#xff09;是保障嵌入式系统可靠性的关键硬件&#xff0c;它通过定期接收 “喂狗” 信号监控系统运行状态&#xff0c;当系统故障…

探索 Vue 3.6 新特性:Vapor Mode 与高性能 Web 应用开发

Vue 3.6 简介 Vue.js 是一个广受欢迎的渐进式 JavaScript 框架&#xff0c;以其简洁的 API、灵活的组件系统和高性能著称。Vue 3.6 是 Vue 3 系列的一个重要版本&#xff0c;引入了多项性能优化和新特性&#xff0c;尤其是备受关注的 Vapor Mode&#xff0c;这是一个无需虚拟 D…

初识prometheus

Prometheus&#xff1a;云原生时代的监控利器 在当今快速发展的云原生和微服务架构时代&#xff0c;传统的监控系统面临着巨大的挑战&#xff1a;如何高效地收集海量、动态变化的指标&#xff1f;如何实时告警并快速定位问题&#xff1f;如何实现灵活的可视化和强大的数据查询…

从源码角度分析导致 JVM 内存泄露的 ThreadLocal

文章目录1. 为什么需要ThreadLocal2. ThreadLocal的实现解析1.1 实现分析1.2 具体实现1.3 ThreadLocalMap中Hash冲突的解决1.3.1 Hash冲突解决的几种方法1.3.1.1 开放定值法1.3.1.2 链地址法1.3.1.3再哈希法&#xff1a;1.3.1.4 建立公共溢出区1.3.2 ThreadLocal解决Hash冲突的…

React组件化的封装

1. 组件化封装的结构 1.1. 定义一个类(组件名必须是大写&#xff0c;小写会被认为是html元素), 继续自React.Component1.2. 实现当前组件的render函数 render当中返回的jsx内容&#xff0c;就是之后React会帮助我们渲染的内容 1.3. 结构图如下&#xff1a; data 方法render()…

嵌入式仿真教学的革新力量:深圳航天科技创新研究院引领高效学习新时代

嵌入式系统作为现代信息技术的核心基石&#xff0c;已深度融入工业控制、物联网、智能终端等关键领域。高校肩负着培养嵌入式技术人才的重任&#xff0c;但传统教学方式正面临严峻挑战&#xff1a;硬件实验设备投入巨大、更新滞后、维护繁琐、时空限制严格&#xff0c;难以满足…

六、Linux核心服务与包管理

作者&#xff1a;IvanCodes 日期&#xff1a;2025年8月3日 专栏&#xff1a;Linux教程 要保证一个Linux系统稳定、安全、功能完备&#xff0c;有效管理其后台服务和软件包是至关重要的。本文将深入介绍现代Linux系统中四个核心的管理工具&#xff1a;systemctl (服务管理)&…

【数据结构】哈希表实现

目录 1. 哈希概念 2 哈希冲突和哈希函数 3. 负载因子 4. 将关键字转为整数 5. 哈希函数 5.1直接定址法 5.2 除法散列法/除留余数法 5.3 乘法散列法&#xff08;了解&#xff09; 5.4 全域散列法&#xff08;了解&#xff09; 5.5 其他方法&#xff08;了解&#xff09…

PostgreSQL面试题及详细答案120道(21-40)

《前后端面试题》专栏集合了前后端各个知识模块的面试题&#xff0c;包括html&#xff0c;javascript&#xff0c;css&#xff0c;vue&#xff0c;react&#xff0c;java&#xff0c;Openlayers&#xff0c;leaflet&#xff0c;cesium&#xff0c;mapboxGL&#xff0c;threejs&…

数据建模及基本数据分析

目录 &#xff08;一&#xff09;数据建模 1.以数据预测为核心的建模 2.以数据聚类为核心的建模 &#xff08;二&#xff09;基本数据分析 1.Numpy 2. Pandas 3.实例 4.Matplotlib 资料自取&#xff1a; 链接: https://pan.baidu.com/s/1PROmz-2hR3VCTd6Eei6lFQ?pwdy8…

电动汽车DCDC转换器的用途及工作原理

在电动汽车的电气架构中&#xff0c;DCDC转换器&#xff08;直流-直流转换器&#xff09;是一个至关重要的部件&#xff0c;负责协调高压动力电池&#xff08;通常300V~800V&#xff09;与低压电气系统&#xff08;12V/24V&#xff09;之间的能量流动。它的性能直接影响整车的能…

PyTorch 应用于3D 点云数据处理汇总和点云配准示例演示

PyTorch 已广泛应用于 3D 点云数据处理&#xff0c;特别是在深度学习驱动的任务中如&#xff1a; 分类、分割、配准、重建、姿态估计、SLAM、目标检测 等。 传统 3D 点云处理以 PCL、Open3D 为主&#xff0c;深度学习方法中&#xff0c;PyTorch 是构建神经网络处理点云的核心框…

ABP VNext + Quartz.NET vs Hangfire:灵活调度与任务管理

ABP VNext Quartz.NET vs Hangfire&#xff1a;灵活调度与任务管理 &#x1f680; &#x1f4da; 目录ABP VNext Quartz.NET vs Hangfire&#xff1a;灵活调度与任务管理 &#x1f680;✨ TL;DR&#x1f6e0; 环境与依赖&#x1f527; Quartz.NET 在 ABP 中接入1. 安装与模块…

[硬件电路-148]:数字电路 - 什么是CMOS电平、TTL电平?还有哪些其他电平标准?发展历史?

1. CMOS电平定义&#xff1a; CMOS&#xff08;Complementary Metal-Oxide-Semiconductor&#xff09;电平基于互补金属氧化物半导体工艺&#xff0c;由PMOS和NMOS晶体管组成。其核心特点是低功耗、高抗干扰性和宽电源电压范围&#xff08;通常为3V~18V&#xff09;。关键参数&…

0基礎網站開發技術教學(二) --(前端篇 2)--

書接上回說到的前端3種主語言以及其用法&#xff0c;這期我們再來探討一下javascript的一些編碼技術。 一) 自定義函數 假如你要使用一個功能&#xff0c;正常來說直接敲出來便可。可如果這個功能你要用不止一次呢?難道你每次都敲出來嗎?這個時侯&#xff0c;就要用到我們的自…

前端 拼多多4399笔试题目

拼多多 3 选择题 opacity|visibity|display区别 在CSS中&#xff0c;opacity: 0 和 visibility: hidden 都可以让元素不可见&#xff0c;但它们的行为不同&#xff1a; ✅ opacity: 0&#xff08;透明度为0&#xff09; 元素仍然占据空间&#xff08;不移除文档流&#xff0…

数琨创享:全球汽车高端制造企业 QMS质量管理平台案例

01.行业领军者的质量升级使命在全球汽车产业链加速升级的浪潮中&#xff0c;质量管控能力已成为企业核心竞争力的关键。作为工信部认证的制造业单项冠军示范企业&#xff0c;万向集团始终以“全球制造、全球市场、做行业领跑者”为战略愿景。面对奔驰、宝马、大众等“9N”高端客…