说一说 LockSupport 的 park 和 unpark

简介: 我是小假 期待与你的下一次相遇 ~

前言

熟悉 Java 并发包的人一定对 LockSupport 的 park/unpark 方法不会感到陌生,它是 Lock(AQS)的基石,给 Lock(AQS)提供了挂起/恢复当前线程的能力。

LockSupport 的 park/unpark 方法本质上是对 Unsafe 的 park/unpark 方法的简单封装,而后者是 native 方法,对 Java 程序来说是一个黑箱操作,那么要想了解它的底层实现,就必须深入 Java 虚拟机的源码。

以park的源码为例:

  1. public class LockSupport {
  2.    public static void park(Object blocker) {
  3.        //获取当前线程
  4.        Thread t = Thread.currentThread();
  5.        //记录当前线程阻塞的原因,底层就是unsafe.putObject,就是把对象存储起来
  6.        setBlocker(t, blocker);
  7.        //执行park
  8.        unsafe.park(false, 0L);
  9.        //线程恢复后,去掉阻塞原因
  10.        setBlocker(t, null);
  11.    }
  12.    //无限阻塞线程,直到有其他线程调用unpark方法
  13.    public static void park() {
  14.        UNSAFE.park(false, 0L);
  15.    }      
  16. }

从源码可以看到真实的实现均在 unsafe。

一、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函数列表

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

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

1.2、基本使用

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

先 park 再 unpark

  1. Thread t1 = new Thread(() -> {
  2.    log.debug("start...");
  3.    sleep(1);
  4.    log.debug("park...");
  5.    LockSupport.park();
  6.    log.debug("resume...");
  7. },"t1");
  8. t1.start();
  9. sleep(2);
  10. log.debug("unpark...");
  11. LockSupport.unpark(t1);

输出:

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

先 unpark 再 park

  1. Thread t1 = new Thread(() -> {
  2.    log.debug("start...");
  3.    sleep(2);
  4.    log.debug("park...");
  5.    LockSupport.park();
  6.    log.debug("resume...");
  7. }, "t1");
  8. t1.start();
  9. sleep(1);
  10. log.debug("unpark...");
  11. LockSupport.unpark(t1);

输出:

  1. 18:43:50.765 c.TestParkUnpark [t1] - start...
  2. 18:43:51.764 c.TestParkUnpark [main] - unpark...
  3. 18:43:52.769 c.TestParkUnpark [t1] - park...
  4. 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一样也能响应中断。

  1. public class LockSupportTest {
  2.    public static void main(String[] args) throws InterruptedException {
  3.        Thread t = new Thread(() -> {
  4.            LockSupport.park();
  5.            System.out.println("thread:"+Thread.currentThread().getName()+"awake");
  6.            },"t1");
  7.        t.start();
  8.        Thread.sleep(2000);
  9.        //中断
  10.        t.interrupt();
  11.    }
  12. }

二、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 变量说明
  1. public class LockSupport {
  2. // Hotspot implementation via intrinsics API
  3. //unsafe常量,设置为使用Unsafe.compareAndSwapInt进行更新
  4. //UNSAFE字段表示sun.misc.Unsafe类,一般程序中不允许直接调用
  5. private static final sun.misc.Unsafe UNSAFE;
  6. //表示parkBlocker在内存地址的偏移量
  7. private static final long parkBlockerOffset;
  8. //表示threadLocalRandomSeed在内存地址的偏移量,此变量的作用暂时还不了解
  9. private static final long SEED;
  10. //表示threadLocalRandomProbe在内存地址的偏移量,此变量的作用暂时还不了解
  11. private static final long PROBE;
  12. //表示threadLocalRandomSecondarySeed在内存地址的偏移量
  13. // 作用是 可以通过nextSecondarySeed()方法来获取随机数
  14. private static final long SECONDARY;
  15. }

  1. 变量是如何获取其实例对象的?
  1. public class LockSupport {
  2. static {
  3.     try {
  4.         //实例化unsafe对象
  5.         UNSAFE = sun.misc.Unsafe.getUnsafe();
  6.         Class<?> tk = Thread.class;
  7.         //利用unsafe对象来获取parkBlocker在内存地址的偏移量
  8.         parkBlockerOffset = UNSAFE.objectFieldOffset(tk.getDeclaredField("parkBlocker"));
  9.         //利用unsafe对象来获取threadLocalRandomSeed在内存地址的偏移量
  10.         SEED = UNSAFE.objectFieldOffset(tk.getDeclaredField("threadLocalRandomSeed"));
  11.         //利用unsafe对象来获取threadLocalRandomProbe在内存地址的偏移量  
  12.         PROBE = UNSAFE.objectFieldOffset(tk.getDeclaredField("threadLocalRandomProbe"));
  13.         //利用unsafe对象来获取threadLocalRandomSecondarySeed在内存地址的偏移量  
  14.         SECONDARY = UNSAFE.objectFieldOffset(tk.getDeclaredField("threadLocalRandomSecondarySeed"));
  15.     } catch (Exception ex) { throw new Error(ex); }
  16. }
  17. }
  1. 由上面代码可知这些变量是通过static代码块在类加载的时候就通过unsafe对象获取其在内存地址的偏移量了。
    3.2 构造方法
  1. public class LockSupport {
  2. //LockSupport只有一个私有构造函数,无法被实例化。
  3. private LockSupport() {} // Cannot be instantiated.
  4. }

  1. 3.3 两个特殊的方法
  1. public class LockSupport {
  2. //设置线程t的parkBlocker字段的值为arg
  3. private static void setBlocker(Thread t, Object arg) {
  4.     // Even though volatile, hotspot doesn't need a write barrier here.
  5.     //尽管hotspot易变,但在这里并不需要写屏障。
  6.     UNSAFE.putObject(t, parkBlockerOffset, arg);
  7. }
  8. //获取当前线程的Blocker值
  9. public static Object getBlocker(Thread t) {
  10.     //若当前线程为空就抛出异常
  11.     if (t == null)
  12.         throw new NullPointerException();
  13.     //利用unsafe对象获取当前线程的Blocker值
  14.     return UNSAFE.getObjectVolatile(t, parkBlockerOffset);
  15. }
  16. }

  1. 3.4 常用方法

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

  1. 2、park(Object blocker)方法 和 park()方法
  1. public class LockSupport {
  2. //阻塞当前线程,并且将当前线程的parkBlocker字段设置为blocker
  3. public static void park(Object blocker) {
  4.     //获取当前线程
  5.     Thread t = Thread.currentThread();
  6.     //将当前线程的parkBlocker字段设置为blocker
  7.     setBlocker(t, blocker);
  8.     //阻塞当前线程,第一个参数表示isAbsolute,是否为绝对时间,第二个参数就是代表时间
  9.     UNSAFE.park(false, 0L);
  10.     //重新可运行后再此设置Blocker
  11.     setBlocker(t, null);
  12. }
  13. //无限阻塞线程,直到有其他线程调用unpark方法
  14. public static void park() {
  15.     UNSAFE.park(false, 0L);
  16. }  
  17. }
  1. 说明:
  • 调用park函数时,首先获取当前线程,然后设置当前线程的parkBlocker字段,即调用setBlocker函数, 之后调用Unsafe类的park函数,之后再调用setBlocker函数。
    park(Object blocker)函数中要调用两次setBlocker函数
  • 1、调用park函数时,当前线程首先设置好parkBlocker字段,然后再调用 Unsafe的park函数,此时,当前线程就已经阻塞了,等待该线程的unpark函数被调用,所以后面的一个 setBlocker函数无法运行,unpark函数被调用,该线程获得许可后,就可以继续运行了,也就运行第二个 setBlocker,把该线程的parkBlocker字段设置为null,这样就完成了整个park函数的逻辑。
  • 2、如果没有第二个 setBlocker,那么之后没有调用park(Object blocker),而直接调用getBlocker函数,得到的还是前一个 park(Object blocker)设置的blocker,显然是不符合逻辑的。总之,必须要保证在park(Object blocker)整个函数 执行完后,该线程的parkBlocker字段又恢复为null。

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

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

  1. public class LockSupport {
  2.    //阻塞当前线程nanos秒
  3.    public static void parkNanos(Object blocker, long nanos) {
  4.        //先判断nanos是否大于0,小于等于0都代表无限等待
  5.        if (nanos > 0) {
  6.            //获取当前线程
  7.            Thread t = Thread.currentThread();
  8.            //将当前线程的parkBlocker字段设置为blocker
  9.            setBlocker(t, blocker);
  10.            //阻塞当前线程现对时间的nanos秒
  11.            UNSAFE.park(false, nanos);
  12.            //将当前线程的parkBlocker字段设置为null
  13.            setBlocker(t, null);
  14.        }
  15.    }  
  16.    //阻塞当前线程nanos秒,现对时间
  17.    public static void parkNanos(long nanos) {
  18.        if (nanos > 0)
  19.            UNSAFE.park(false, nanos);
  20.    }  
  21. }

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

  1. public class LockSupport {
  2.    //将当前线程阻塞绝对时间的deadline秒,并且将当前线程的parkBlockerOffset设置为blocker
  3.    public static void parkUntil(Object blocker, long deadline) {
  4.        //获取当前线程
  5.        Thread t = Thread.currentThread();
  6.        //设置当前线程parkBlocker字段设置为blocker
  7.        setBlocker(t, blocker);
  8.        //阻塞当前线程绝对时间的deadline秒
  9.        UNSAFE.park(true, deadline);
  10.        //当前线程parkBlocker字段设置为null
  11.        setBlocker(t, null);
  12.    }
  13.    //将当前线程阻塞绝对时间的deadline秒
  14.    public static void parkUntil(long deadline) {
  15.        UNSAFE.park(true, deadline);
  16.    }  
  17. }

总结:

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

相关文章
|
9月前
|
Arthas 监控 Java
Java死锁 如何定位?如何避免Java死锁?(图解+秒懂+史上最全)
Java死锁 如何定位?如何避免Java死锁?(图解+秒懂+史上最全)
Java死锁 如何定位?如何避免Java死锁?(图解+秒懂+史上最全)
|
9月前
|
Java API 调度
我们来说说 LockSupport 的 park 和 unpark
我是小假 期待与你的下一次相遇 ~
142 1
|
10月前
|
JSON IDE Java
20 款 IDEA 主题任你选!(快来看看你最喜欢那个~)
我是小假 期待与你的下一次相遇 ~
5510 1
|
10月前
|
算法 Java 索引
说一说 Java 并发队列原理剖析
我是小假 期待与你的下一次相遇 ~
113 1
|
算法 Java 调度
Semaphore实现原理全面解析
Semaphore(信号量)是一个同步工具类,通过Semaphore可以控制同时访问共享资源的线程个数。
|
JSON 前端开发 安全
【潜意识java】前后端跨域问题及解决方案
本文深入探讨了跨域问题及其解决方案。跨域是指浏览器出于安全考虑,限制从一个域加载的网页请求另一个域的资源。
4147 0
|
消息中间件 负载均衡 算法
聊聊 RocketMQ中 Topic,Queue,Consumer,Consumer Group的关系
本文详细解析了RocketMQ中Topic、Queue、Consumer及Consumer Group之间的关系。文中通过图表展示了Topic可包含多个Queue,Queue分布在不同Broker上;Consumer组内多个消费者共享消息;并深入探讨了集群消费与广播消费模式下Queue与Consumer的关系,以及Rebalancing机制在实例增减时如何确保负载均衡。理解这些关系有助于更好地掌握RocketMQ的工作原理,提升系统运维效率。
3449 2
|
安全 Linux 网络安全
【工具使用】几款优秀的SSH连接客户端软件工具推荐FinalShell、Xshell、MobaXterm、OpenSSH、PUTTY、Terminus、mRemoteNG、Terminals等
【工具使用】几款优秀的SSH连接客户端软件工具推荐FinalShell、Xshell、MobaXterm、OpenSSH、PUTTY、Terminus、mRemoteNG、Terminals等
142539 0
springboot静态资源目录访问,及自定义静态资源路径,index页面的访问
本文介绍了Spring Boot中静态资源的访问位置、如何进行静态资源访问测试、自定义静态资源路径和静态资源请求映射,以及如何处理自定义静态资源映射对index页面访问的影响。提供了两种解决方案:取消自定义静态资源映射或编写Controller来截获index.html的请求并重定向。
springboot静态资源目录访问,及自定义静态资源路径,index页面的访问