【JUC】十六、LockSupport类实现线程等待与唤醒

2023-11-30 06:36

本文主要是介绍【JUC】十六、LockSupport类实现线程等待与唤醒,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 1、LockSupport
  • 2、wait和notify存在的问题
  • 3、await和signal存在的问题
  • 4、park和unpark方法
  • 5、LockSupport用法示例
  • 6、Permit不会累积
  • 7、面试

在这里插入图片描述

1、LockSupport

线程等待和唤醒的方式有:

  • 使用Object的wait方法让对象上活动的线程等待,使用notify方法来唤醒线程
  • 使用JUC报中Condition的await方法让线程等待,使用signal方法来唤醒线程
  • LockSupport类来阻塞当前线程以及唤醒指定被阻塞的线程

在这里插入图片描述

LockSupport是一个线程阻塞工具类,所有方法都是静态的。可以让线程在任意位置阻塞,阻塞之后也有对应的唤醒方法,归根结底,LockSupport调用的Unsafe中的native代码。

  • LockSupport提供park和unpark方法实现阻塞线程和解除阻塞
  • LockSupport和每个使用它的线程都有一个许可permit关联
  • 每个线程都有一个相关的permit,permit最多只有一个,重复调用unpark也不会积累凭证
  • LockSupport就是通过Permit (许可)的概念来做到阻塞和唤阻线程的功能

2、wait和notify存在的问题

正常使用wait和notify如下:

public class LockSupport1 {public static void main(String[] args) {Object o = new Object();new Thread(() -> {synchronized (o){System.out.println(Thread.currentThread().getName() + "\t come in...");try {o.wait();} catch (InterruptedException e) {e.printStackTrace();}System.out.println(Thread.currentThread().getName() + "\t 被唤醒");}},"t1").start();//歇200mstry { TimeUnit.MILLISECONDS.sleep(200); } catch (InterruptedException e) { e.printStackTrace(); }new Thread(() -> {synchronized (o){System.out.println(Thread.currentThread().getName() + "\t come in ...");o.notify();System.out.println(Thread.currentThread().getName() + "\t 发出唤醒通知");}},"t2").start();}
}

此时一切正常:

在这里插入图片描述

问题1:去掉syncyronized

在这里插入图片描述

问题2:先唤醒再等待

在这里插入图片描述

可以发现,将notify先于wait执行,等待被唤醒的线程会陷入无限等待中,就像叫你起床的人,先走了,你睡着以后没人再叫你了。

问题点总结:

  • wait和notify方法必须要在同步块或者同步方法里面,且成对出现和使用
  • 必须先wait再notify,反之唤醒失败

3、await和signal存在的问题

常规用法:

public class LockSupport1 {public static void main(String[] args) {Lock lock = new ReentrantLock();Condition condition = lock.newCondition();new Thread(() -> {lock.lock();System.out.println(Thread.currentThread().getName() + "\t come in...");try {condition.await();} catch (InterruptedException e) {e.printStackTrace();} finally {lock.unlock();}System.out.println(Thread.currentThread().getName() + "\t 被唤醒");},"t1").start();//歇200mstry { TimeUnit.MILLISECONDS.sleep(200); } catch (InterruptedException e) { e.printStackTrace(); }new Thread(() -> {lock.lock();try {System.out.println(Thread.currentThread().getName() + "\t come in...");condition.signal();System.out.println(Thread.currentThread().getName() + "\t 发出唤醒通知");} finally {lock.unlock();}},"t2").start();}}

在这里插入图片描述

问题1:去掉锁

在这里插入图片描述

可以看到去掉lock和unlock加解锁,await和signal都触发了IllegalMonitorStateException异常。

异常2:先唤醒再等待

在这里插入图片描述

可以看到,前两种线程等待和唤醒的方式,都有使用限制:

  • 线程必须先持有锁(synchronized或者lock)
  • 必须先等待后唤醒,才能唤醒成功

4、park和unpark方法

基于前面两种方式的缺陷,LockSupport提供了新的实现思路来解决 ⇒凭证 。通过park()和unpark(thread)来实现阻塞和唤醒线程。

//阻塞当前线程
park()
//阻塞传入的具体线程
park(Thread thread)

当调用park方法时:

  • 如果线程有凭证,则直接消耗掉这个凭证然后正常往下执行
  • 如果线程没有凭证,就必须阻塞等待到凭证可用

看下源码,第二个参数就是用来指定多久放行的,默认0,即没许可证不放行,直到别的线程给当前线程发放permit:

在这里插入图片描述

unpark(Thread thread)

当调用unpark方法时:

  • 给传入的线程发一个凭证
  • 自动唤醒之前阻塞的LockSupport.park()
  • 但凭证最多只有一个,多次调用不会累加

5、LockSupport用法示例

开两个线程t2给t1发通行证:

public class LockSupport2 {public static void main(String[] args) {Thread t1 = new Thread(() -> {System.out.println(Thread.currentThread().getName() + "\t ---com in: " + System.currentTimeMillis());LockSupport.park();System.out.println(Thread.currentThread().getName() + "\t ---被唤醒: " + System.currentTimeMillis());}, "t1");t1.start();try { TimeUnit.MILLISECONDS.sleep(200); } catch (InterruptedException e) { e.printStackTrace(); }new Thread(() -> {System.out.println(Thread.currentThread().getName() + "\t ---com in");System.out.println(Thread.currentThread().getName() + "给t1线程发了permit");LockSupport.unpark(t1);},"t2").start();}
}

等待唤醒成功,可以发现不用锁块了,也没有synchronized或者lock了。

在这里插入图片描述

再看先发许可证会不会被成功唤醒:

public class LockSupport2 {public static void main(String[] args) {Thread t1 = new Thread(() -> {//休息两秒,让t2先执行try { TimeUnit.MILLISECONDS.sleep(2000); } catch (InterruptedException e) { e.printStackTrace(); }System.out.println(Thread.currentThread().getName() + "\t ---com in: " + System.currentTimeMillis());LockSupport.park();System.out.println(Thread.currentThread().getName() + "\t ---被唤醒: " + System.currentTimeMillis());}, "t1");t1.start();//try { TimeUnit.MILLISECONDS.sleep(200); } catch (InterruptedException e) { e.printStackTrace(); }new Thread(() -> {System.out.println(Thread.currentThread().getName() + "\t ---com in");System.out.println(Thread.currentThread().getName() + "给t1线程发了permit");LockSupport.unpark(t1);},"t2").start();}
}

在这里插入图片描述

先发permit,再LockSupport.park(),就像持证上岗,或者高速公路的ETC,提前缴费买了通行证后走高速,遇到关卡一路通畅,此时park形同虚设

6、Permit不会累积

验证一个线程醉倒一个Permit,许可证不会累积:

public class LockSupport3 {public static void main(String[] args) {Thread t1 = new Thread(() -> {//先让发许可证的线程执行try { TimeUnit.MILLISECONDS.sleep(2000); } catch (InterruptedException e) { e.printStackTrace(); }System.out.println(Thread.currentThread().getName() + "\t ---come in");LockSupport.park();LockSupport.park();System.out.println(Thread.currentThread().getName() + "\t ---被唤醒");}, "t1");t1.start();new Thread(() -> {LockSupport.unpark(t1);LockSupport.unpark(t1);LockSupport.unpark(t1);LockSupport.unpark(t1);}).start();}
}

可以看到多次unpark也不能过两个park:

在这里插入图片描述

稍微再改一下,一个凭证用完后,自己再给自己发一个,就可以通过了。

在这里插入图片描述

7、面试

Q1:为什么LockSupport可以突破wait/notify原有的调用顺序限制?

A1:因为unpark后线程t获得了一个凭证,之后线程t再调用park,就凭证消费,畅通无阻

Q2:为什么唤醒两次后再阻塞两次,最终结果还是阻塞?

A2:因为凭证的数量最多为1,连续调用两次unpark并不会有两个凭证,而调用两次park却要消耗两个凭证,凭证不够,不能放行。

这篇关于【JUC】十六、LockSupport类实现线程等待与唤醒的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Flutter实现文字镂空效果的详细步骤

《Flutter实现文字镂空效果的详细步骤》:本文主要介绍如何使用Flutter实现文字镂空效果,包括创建基础应用结构、实现自定义绘制器、构建UI界面以及实现颜色选择按钮等步骤,并详细解析了混合模... 目录引言实现原理开始实现步骤1:创建基础应用结构步骤2:创建主屏幕步骤3:实现自定义绘制器步骤4:构建U

SpringBoot中四种AOP实战应用场景及代码实现

《SpringBoot中四种AOP实战应用场景及代码实现》面向切面编程(AOP)是Spring框架的核心功能之一,它通过预编译和运行期动态代理实现程序功能的统一维护,在SpringBoot应用中,AO... 目录引言场景一:日志记录与性能监控业务需求实现方案使用示例扩展:MDC实现请求跟踪场景二:权限控制与

Android实现定时任务的几种方式汇总(附源码)

《Android实现定时任务的几种方式汇总(附源码)》在Android应用中,定时任务(ScheduledTask)的需求几乎无处不在:从定时刷新数据、定时备份、定时推送通知,到夜间静默下载、循环执行... 目录一、项目介绍1. 背景与意义二、相关基础知识与系统约束三、方案一:Handler.postDel

使用Python实现IP地址和端口状态检测与监控

《使用Python实现IP地址和端口状态检测与监控》在网络运维和服务器管理中,IP地址和端口的可用性监控是保障业务连续性的基础需求,本文将带你用Python从零打造一个高可用IP监控系统,感兴趣的小伙... 目录概述:为什么需要IP监控系统使用步骤说明1. 环境准备2. 系统部署3. 核心功能配置系统效果展

Python实现微信自动锁定工具

《Python实现微信自动锁定工具》在数字化办公时代,微信已成为职场沟通的重要工具,但临时离开时忘记锁屏可能导致敏感信息泄露,下面我们就来看看如何使用Python打造一个微信自动锁定工具吧... 目录引言:当微信隐私遇到自动化守护效果展示核心功能全景图技术亮点深度解析1. 无操作检测引擎2. 微信路径智能获

Python中pywin32 常用窗口操作的实现

《Python中pywin32常用窗口操作的实现》本文主要介绍了Python中pywin32常用窗口操作的实现,pywin32主要的作用是供Python开发者快速调用WindowsAPI的一个... 目录获取窗口句柄获取最前端窗口句柄获取指定坐标处的窗口根据窗口的完整标题匹配获取句柄根据窗口的类别匹配获取句

在 Spring Boot 中实现异常处理最佳实践

《在SpringBoot中实现异常处理最佳实践》本文介绍如何在SpringBoot中实现异常处理,涵盖核心概念、实现方法、与先前查询的集成、性能分析、常见问题和最佳实践,感兴趣的朋友一起看看吧... 目录一、Spring Boot 异常处理的背景与核心概念1.1 为什么需要异常处理?1.2 Spring B

Python位移操作和位运算的实现示例

《Python位移操作和位运算的实现示例》本文主要介绍了Python位移操作和位运算的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一... 目录1. 位移操作1.1 左移操作 (<<)1.2 右移操作 (>>)注意事项:2. 位运算2.1

如何在 Spring Boot 中实现 FreeMarker 模板

《如何在SpringBoot中实现FreeMarker模板》FreeMarker是一种功能强大、轻量级的模板引擎,用于在Java应用中生成动态文本输出(如HTML、XML、邮件内容等),本文... 目录什么是 FreeMarker 模板?在 Spring Boot 中实现 FreeMarker 模板1. 环

Qt实现网络数据解析的方法总结

《Qt实现网络数据解析的方法总结》在Qt中解析网络数据通常涉及接收原始字节流,并将其转换为有意义的应用层数据,这篇文章为大家介绍了详细步骤和示例,感兴趣的小伙伴可以了解下... 目录1. 网络数据接收2. 缓冲区管理(处理粘包/拆包)3. 常见数据格式解析3.1 jsON解析3.2 XML解析3.3 自定义