Java并发包源码学习系列:挂起与唤醒线程LockSupport工具类

2024-01-08 02:58

本文主要是介绍Java并发包源码学习系列:挂起与唤醒线程LockSupport工具类,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

    • LockSupport概述
    • park与unpark相关方法
    • 中断演示
    • blocker的作用
      • 测试无blocker
      • 测试带blocker
    • JDK提供的demo
    • 总结
    • 参考阅读

系列传送门:

  • Java并发包源码学习系列:AbstractQueuedSynchronizer
  • Java并发包源码学习系列:CLH同步队列及同步资源获取与释放
  • Java并发包源码学习系列:AQS共享式与独占式获取与释放资源的区别
  • Java并发包源码学习系列:ReentrantLock可重入独占锁详解
  • Java并发包源码学习系列:ReentrantReadWriteLock读写锁解析
  • Java并发包源码学习系列:详解Condition条件队列、signal和await

LockSupport概述

LockSupport工具类定义了一组公共的静态方法,提供了最基本的线程阻塞和唤醒功能,是创建锁和其他同步类的基础,你会发现,AQS中阻塞线程和唤醒线程的地方,就是使用LockSupport提供的park和unpark方法,比如下面这段:

    // 挂起线程private final boolean parkAndCheckInterrupt() {LockSupport.park(this);return Thread.interrupted();}// 唤醒线程private void unparkSuccessor(Node node) {//...if (s != null)LockSupport.unpark(s.thread);}

park与unpark相关方法

LockSupport提供了一组park开头的方法来阻塞当前线程【省略static】:

  • void park():阻塞当前线程,如果调用unpark(Thread thread)方法或者当前线程被中断,才能从park()方法返回。
  • void parkNanos(long nanos):阻塞当前线程,最长不超过nanos纳秒,返回条件在park()的基础上增加了超时返回。
  • void parkUntil(long deadline):阻塞当前线程,直到deadline【从1970年开始到deadline时间的毫秒数】时间。
  • void unpark(Thread thread):唤醒处于阻塞状态的线程thread。

JDK1.6中,增加了带有blocker参数的几个方法,blocker参数用来标识当前线程在等待的对象,用于问题排查和系统监控。

下面演示park()方法和unpark()方法的使用:

在thread线程中调用park()方法,默认情况下该线程是不持有许可证的,因此将会被阻塞挂起。

unpark(thread)方法将会让thread线程获得许可证,才能从park()方法返回。

    public static void main(String[] args) throws InterruptedException {Thread thread = new Thread(() ->{String name = Thread.currentThread().getName();System.out.println(name + " begin park");LockSupport.park();// 如果调用park的线程已经获得了关联的许可证,就会立即返回System.out.println(name + " end park");},"A");thread.start(); // 默认情况下,thread不持有许可证,会被阻塞挂起Thread.sleep(1000); System.out.println(thread.getName() + " begin unpark");LockSupport.unpark(thread);//让thread获得许可证}
// 结果如下
A begin park
A begin unpark
A end park

你需要理解,许可证在这里的作用,我们也可以事先给线程一个许可证,接着在park的时候就不会被阻塞了。

    public static void main(String[] args) {System.out.println("begin park");// 使当前线程获得许可证LockSupport.unpark(Thread.currentThread());// 再次调用park方法,因为已经有许可证了,不会被阻塞LockSupport.park();System.out.println("end park");}
// 结果如下
begin park
end park

中断演示

线程被中断的时候,park方法不会抛出异常,因此需要park退出之后,对中断状态进行处理。

    public static void main(String[] args) throws InterruptedException {Thread thread = new Thread(() -> {String name = Thread.currentThread().getName();System.out.println(name + " begin park");// 一直挂起自己,只有被中断,才会推出循环while (!Thread.currentThread().isInterrupted()) {LockSupport.park();}System.out.println(name + " end park");}, "A");thread.start();Thread.sleep(1000);System.out.println("主线程准备中断线程" + thread.getName());// 中断threadthread.interrupt();}
// 结果如下
A begin park
主线程准备中断线程A
A end park

blocker的作用

JDK1.6开始,一系列park方法开始支持传入blocker参数,标识当前线程在等待的对象,当线程在没有持有许可证的情况下调用park方法而被阻塞挂起时,这个blocker对象会被记录到该线程内部。

    public static void park(Object blocker) {Thread t = Thread.currentThread();setBlocker(t, blocker); // 设置blockerUNSAFE.park(false, 0L);setBlocker(t, null); // 清除blocker}

Thread类里有个volatile Object parkBlocker变量,用来存放park方法传递的blocker对象,也就是把blocker变量存放到了调用park方法的线程的成员变量中。

接下来我们通过两个例子感受一下:

测试无blocker

public class TestParkWithoutBlocker {public void park(){LockSupport.park();}public static void main(String[] args) throws InterruptedException {new TestParkWithoutBlocker().park();Thread.sleep(3000);}
}

使用jps命令,列出当前运行的进程4412 TestPark,接着使用jstack 4412命令查看线程堆栈:

测试带blocker

public class TestBlockerPark {public void park(){LockSupport.park(this); // 传入blocker = this}public static void main(String[] args) throws InterruptedException {new TestBlockerPark().park();Thread.sleep(3000);}
}

明显的差别就在于,使用带blocker 参数的park方法,能够通过jstack看到具体阻塞对象的信息:

- parking to wait for  <0x000000076b77dff0> (a chapter6_1_LockSupport.TestBlockerPark)

诊断工具可以调用getBlocker(Thread)方法来获取blocker对象,JDK推荐我们使用带有blocker参数的park方法,并且设置blocker为this,这样当在打印线程堆栈排查问题的时候就能够知道那个类被阻塞了。

JDK提供的demo

老传统了,摘一段JavaDoc上的使用案例:

/*** 先进先出的锁,只有队列的首元素可以获取锁*/
class FIFOMutex {private final AtomicBoolean locked = new AtomicBoolean(false);private final Queue<Thread> waiters= new ConcurrentLinkedQueue<Thread>();public void lock() {// 中断标志boolean wasInterrupted = false; Thread current = Thread.currentThread();waiters.add(current);// 不是队首线程 或 当前锁已经被其他线程获取,则调用park方法挂起自己while (waiters.peek() != current ||!locked.compareAndSet(false, true)) {LockSupport.park(this);// 如果park方法是因为被中断而返回,则忽略中断,并且重置中断标志// 接着再次进入循环if (Thread.interrupted()) // ignore interrupts while waitingwasInterrupted = true;}waiters.remove();// 如果标记为true,则中断线程// [虽然我对中断信号不感兴趣,忽略它,但是不代表其他线程对该标志不感兴趣,因此恢复一下.]if (wasInterrupted)          // reassert interrupt status on exitcurrent.interrupt();}public void unlock() {locked.set(false);LockSupport.unpark(waiters.peek());}
}

总结

  • LockSupport提供了有关线程挂起park和唤醒unpark的静态方法。
  • JDK1.6之后允许传入blocker阻塞对象,便于问题监控和排查。
  • 如果park的线程被中断,不会抛出异常,需要自行对中断状态进行处理。

参考阅读

  • 翟陆续 薛冰田 《Java并发编程之美》
  • 方腾飞 《Java并发编程的艺术》
  • 【J.U.C】LockSupport

这篇关于Java并发包源码学习系列:挂起与唤醒线程LockSupport工具类的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/582117

相关文章

Java中流式并行操作parallelStream的原理和使用方法

《Java中流式并行操作parallelStream的原理和使用方法》本文详细介绍了Java中的并行流(parallelStream)的原理、正确使用方法以及在实际业务中的应用案例,并指出在使用并行流... 目录Java中流式并行操作parallelStream0. 问题的产生1. 什么是parallelS

Java中Redisson 的原理深度解析

《Java中Redisson的原理深度解析》Redisson是一个高性能的Redis客户端,它通过将Redis数据结构映射为Java对象和分布式对象,实现了在Java应用中方便地使用Redis,本文... 目录前言一、核心设计理念二、核心架构与通信层1. 基于 Netty 的异步非阻塞通信2. 编解码器三、

SpringBoot基于注解实现数据库字段回填的完整方案

《SpringBoot基于注解实现数据库字段回填的完整方案》这篇文章主要为大家详细介绍了SpringBoot如何基于注解实现数据库字段回填的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以了解... 目录数据库表pom.XMLRelationFieldRelationFieldMapping基础的一些代

一篇文章彻底搞懂macOS如何决定java环境

《一篇文章彻底搞懂macOS如何决定java环境》MacOS作为一个功能强大的操作系统,为开发者提供了丰富的开发工具和框架,下面:本文主要介绍macOS如何决定java环境的相关资料,文中通过代码... 目录方法一:使用 which命令方法二:使用 Java_home工具(Apple 官方推荐)那问题来了,

JDK21对虚拟线程的几种用法实践指南

《JDK21对虚拟线程的几种用法实践指南》虚拟线程是Java中的一种轻量级线程,由JVM管理,特别适合于I/O密集型任务,:本文主要介绍JDK21对虚拟线程的几种用法,文中通过代码介绍的非常详细,... 目录一、参考官方文档二、什么是虚拟线程三、几种用法1、Thread.ofVirtual().start(

Java HashMap的底层实现原理深度解析

《JavaHashMap的底层实现原理深度解析》HashMap基于数组+链表+红黑树结构,通过哈希算法和扩容机制优化性能,负载因子与树化阈值平衡效率,是Java开发必备的高效数据结构,本文给大家介绍... 目录一、概述:HashMap的宏观结构二、核心数据结构解析1. 数组(桶数组)2. 链表节点(Node

Java AOP面向切面编程的概念和实现方式

《JavaAOP面向切面编程的概念和实现方式》AOP是面向切面编程,通过动态代理将横切关注点(如日志、事务)与核心业务逻辑分离,提升代码复用性和可维护性,本文给大家介绍JavaAOP面向切面编程的概... 目录一、AOP 是什么?二、AOP 的核心概念与实现方式核心概念实现方式三、Spring AOP 的关

详解SpringBoot+Ehcache使用示例

《详解SpringBoot+Ehcache使用示例》本文介绍了SpringBoot中配置Ehcache、自定义get/set方式,并实际使用缓存的过程,文中通过示例代码介绍的非常详细,对大家的学习或者... 目录摘要概念内存与磁盘持久化存储:配置灵活性:编码示例引入依赖:配置ehcache.XML文件:配置

Java 虚拟线程的创建与使用深度解析

《Java虚拟线程的创建与使用深度解析》虚拟线程是Java19中以预览特性形式引入,Java21起正式发布的轻量级线程,本文给大家介绍Java虚拟线程的创建与使用,感兴趣的朋友一起看看吧... 目录一、虚拟线程简介1.1 什么是虚拟线程?1.2 为什么需要虚拟线程?二、虚拟线程与平台线程对比代码对比示例:三

Java中的.close()举例详解

《Java中的.close()举例详解》.close()方法只适用于通过window.open()打开的弹出窗口,对于浏览器的主窗口,如果没有得到用户允许是不能关闭的,:本文主要介绍Java中的.... 目录当你遇到以下三种情况时,一定要记得使用 .close():用法作用举例如何判断代码中的 input