JAVA Thread.yield()方法(请求释放CPU)的理解及在ForkJoinPool中的使用

目录

  • 线程释放资源的三种方法
    • Object.wait()
    • Thread.sleep()
    • Thread.yield()
  • Thread.yield()的特性
  • 使用Thread.yield()测试高负载场景下的多线程协作
    • 测试结果
      • 不调用yield()方法
      • 调用yield()方法
    • 测试结论
  • Thread.yield()在ForkJoinPool中的应用
    • ForkJoinPool的工作原理

线程释放资源的三种方法

系统内部开销:yield < sleep < wait

Object.wait()

wait()方法是一个成员方法,用于主动放弃当前线程获得的对象锁,进入阻塞态,并等待锁对象上的notify()/notifyAll()方法调用,以唤起当前阻塞的线程。
当前线程被唤起时,需要再次尝试获取同步对象的监听器(锁),以能够进入继续执行同步代码块,因此wait()方法被在同步代码块中被调用。
时间开销:线程状态切换时间 + 线程等待时间 + 线程调度时间 + 对象锁获取时间

    /*** Causes the current thread to wait until another thread invokes the* {@link java.lang.Object#notify()} method or the* {@link java.lang.Object#notifyAll()} method for this object.* In other words, this method behaves exactly as if it simply* performs the call {@code wait(0)}.* <p>* The current thread must own this object's monitor. The thread* releases ownership of this monitor and waits until another thread* notifies threads waiting on this object's monitor to wake up* either through a call to the {@code notify} method or the* {@code notifyAll} method. The thread then waits until it can* re-obtain ownership of the monitor and resumes execution.* <p>* As in the one argument version, interrupts and spurious wakeups are* possible, and this method should always be used in a loop:* <pre>*     synchronized (obj) {*         while (&lt;condition does not hold&gt;)*             obj.wait();*         ... // Perform action appropriate to condition*     }* </pre>* This method should only be called by a thread that is the owner* of this object's monitor. See the {@code notify} method for a* description of the ways in which a thread can become the owner of* a monitor.** @throws  IllegalMonitorStateException  if the current thread is not*               the owner of the object's monitor.* @throws  InterruptedException if any thread interrupted the*             current thread before or while the current thread*             was waiting for a notification.  The <i>interrupted*             status</i> of the current thread is cleared when*             this exception is thrown.* @see        java.lang.Object#notify()* @see        java.lang.Object#notifyAll()*/public final void wait() throws InterruptedException {wait(0);}

Thread.sleep()

sleep()方法是一个静态成员方法,调用时主动挂起当前线程,线程不会赶往阻塞态,也不会释放对象监听器,不过在调用此方法时需要指定睡眠时间,以使用线程调度器能够在指定的时间再次唤起当前线程,继续从挂载点执行。
时间开销:线程等待时间 + 线程调度时间

    /*** Causes the currently executing thread to sleep (temporarily cease* execution) for the specified number of milliseconds, subject to* the precision and accuracy of system timers and schedulers. The thread* does not lose ownership of any monitors.** @param  millis*         the length of time to sleep in milliseconds** @throws  IllegalArgumentException*          if the value of {@code millis} is negative** @throws  InterruptedException*          if any thread has interrupted the current thread. The*          <i>interrupted status</i> of the current thread is*          cleared when this exception is thrown.*/public static native void sleep(long millis) throws InterruptedException;

Thread.yield()

yield()方法是一个静态成员方法,会向调度器发送一个请求,表示愿意主动释放自己占用的处理器,避免自己过度占用,导致其它线程饥饿,但这并不意味着当前线程一定会释放资源,这取决于调度器的运行时调度策略,因此此方法并非用于资源同步目的,而是为了提升多线程场景下的处理性能。
调用此方法后,当前线程只会主动释放占用的CPU,而会使当前线程从运行态(RUNNING)转换到就绪态(READY)且不会释放对象锁,同时也不会保证一定能够释放处理器。
时间开销: 线程调度时间

    /*** A hint to the scheduler that the current thread is willing to yield* its current use of a processor. The scheduler is free to ignore this* hint.** <p> Yield is a heuristic attempt to improve relative progression* between threads that would otherwise over-utilise a CPU. Its use* should be combined with detailed profiling and benchmarking to* ensure that it actually has the desired effect.** <p> It is rarely appropriate to use this method. It may be useful* for debugging or testing purposes, where it may help to reproduce* bugs due to race conditions. It may also be useful when designing* concurrency control constructs such as the ones in the* {@link java.util.concurrent.locks} package.*/public static native void yield();

Thread.yield()的特性

  1. 相比于sleep/wail方法,此方法的副作用最小,只会影响CPU时间片的分配;
  2. 适用于计算密集型的工作场景,当资源充足时,收益并不明显或没有;
  3. 不是用于线程同步目的,而是为了协调多线程的工作进程,减缓当前线程对于CPU的长期占用。
  4. 程序设计时,通常结合CAS方法使用,避免线程阻塞或等待;
  5. 调用方法,并不意味着当前线程一定会释放处理器,这取决于调度器。

使用Thread.yield()测试高负载场景下的多线程协作

测试代码的设计思想:

  1. 获取当前系统可用的处理器数量,这里为10;
  2. 创建10个Looping线程,尽最大努力占用处理器时间片,不过其中一个线程,t1需要调用yield()方法,以测试方法调用前后的效果。
  3. 额外创建一个工作线程t2,以验证yield()方法调用时,这个线程确实受到了影响。

为了能够方便观测t2线程确实受到了影响,即yield()方法确实提升了多线程的相对工作进度,这里在t1线程looping时进行计数,以查看t2线程执行时的数字大小,从而判定影响。

  public statick void main() {var sync = 0var t2Run = falsevar quit = false// 10 processorsvar processors = Runtime.getRuntime.availableProcessors()println("Available processors: " + processors)// make 1 thread to loop and to yield after counted 8,000,000val t1 = new Thread {override def run(): Unit = {while (!quit) {if (sync > 8000000) {println(s"${Thread.currentThread().getName} - hopefully t2 to work $sync")t2Run = true// Key pointThread.`yield`()}sync += 1}}}t1.start()// make 8 threads to loopingwhile (processors > 1) {val t = new Thread {override def run(): Unit = {while(true) {// busy cpu}}}t.start()processors -= 1}Thread.sleep(2000)val t2 = new Thread {override def run(): Unit = {while (!t2Run) {// looping to wait breaking}println(s"${Thread.currentThread().getName} - working with $t2Run")quit = trueprintln(s"${Thread.currentThread().getName} - working done")System.exit(0)}}t2.start()while (true) {// main thread looping}}

测试结果

重点观测计数器的显示值,以判定yield()方法调用前后的差别。

不调用yield()方法

实验一:
Thread-14 - hopefully t2 working 8496165
Thread-14 - hopefully t2 working 8496166
Thread-14 - hopefully t2 working 8496167
Thread-24 - working with true
Thread-24 - working done
Thread-14 - hopefully t2 working 8496168
实验二:
Thread-14 - hopefully t2 working 8535080
Thread-14 - hopefully t2 working 8535081
Thread-14 - hopefully t2 working 8535082
Thread-24 - working with true
Thread-14 - hopefully t2 working 8535083
Thread-24 - working done
实验三:
Thread-14 - hopefully t2 working 8633271
Thread-14 - hopefully t2 working 8633272
Thread-14 - hopefully t2 working 8633273
Thread-24 - working with true
Thread-24 - working done
Thread-14 - hopefully t2 working 8633274

调用yield()方法

实验一:
Thread-14 - hopefully t2 working 8092040
Thread-14 - hopefully t2 working 8092041
Thread-14 - hopefully t2 working 8092042
Thread-24 - working with true
Thread-24 - working done
实验二:
Thread-14 - hopefully t2 working 8127893
Thread-14 - hopefully t2 working 8127894
Thread-14 - hopefully t2 working 8127895
Thread-24 - working with true
Thread-24 - working done
实验三:
Thread-14 - hopefully t2 working 8004998
Thread-14 - hopefully t2 working 8004999
Thread-14 - hopefully t2 working 8004999
Thread-24 - working with true
Thread-24 - working done

测试结论

经过多轮实验,大部分情况下,调用yield()方法时,t2线程的执行时间提前了(计数器最大8128893),也意味着整个JVM进程会提前退出;而不调用时,t2线程的执行触发时间被延迟了,计数器最大为8633271

由此看,yield()方法确实能够达到主动释放处理器/cpu的目的,但释放时机是不确定的,同时是否一定会释放取决于调度器的实时决策。
yield方法能够达到协调多线程的相对处理进度,通过线程主动释放正在占用的CPU时间片,从一定上提升了系统的整体性能,但具体是能够提升多少需要根据实际场景进行充分的测试。

Thread.yield()在ForkJoinPool中的应用

ForkJoinPool是在JDK 1.7版本引入的,采用分治方法,将一个大任务拆分成多个子任务并行计算再聚合的工作模式,适用于计算密集型的工作场景,它的主要特性如下:

  1. ForkJoinPool是一个使用分治策略递归执行任务的线程池。
  2. 它被诸如Kotlin和Akka之类的JVM语言用来构建消息驱动的应用程序。
  3. ForkJoinPool并行执行任务,从而实现计算机资源的高效使用。
  4. 工作窃取算法通过允许空闲线程从繁忙线程中窃取任务来优化资源利用率。
  5. 任务存储在一个双端队列中,LIFO策略用于存储,FIFO用于窃取。
  6. ForkJoinPool框架中的主要类包括ForkJoinPool、RecursiveAction和RecursiveTask:
  7. RecursiveAction用于计算递归操作,不返回任何值。
  8. RecursiveTask类似,但返回一个值。
  9. compute()方法在两个类中都被重写以实现自定义逻辑。
  10. fork()方法调用compute()方法并将任务分解为更小的子任务。
  11. join()方法等待子任务完成并合并它们的结果。
  12. ForkJoinPool通常与并行流和CompletableFuture一起使用。

awaitQuiescence(...)方法旨在使线程池尽快沉寂,即利用work-stealing机制尽快完成任务,因此内部会使用while循环不断地查检队列中是否还有空闲线程能够帮忙处理任务,因此为了避免在无可用资源的情况下,一直消耗CPU资源同时避免线程阻塞,在方法实现时调用了Thread.yield()方法,试图在每一次检查时都主动释放CPU时间片,以尝试加快其它线程 的进度 。

public class ForkJoinPool extends AbstractExecutorService {/*** If called by a ForkJoinTask operating in this pool, equivalent* in effect to {@link ForkJoinTask#helpQuiesce}. Otherwise,* waits and/or attempts to assist performing tasks until this* pool {@link #isQuiescent} or the indicated timeout elapses.** @param timeout the maximum time to wait* @param unit the time unit of the timeout argument* @return {@code true} if quiescent; {@code false} if the* timeout elapsed.*/public boolean awaitQuiescence(long timeout, TimeUnit unit) {long nanos = unit.toNanos(timeout);ForkJoinWorkerThread wt;Thread thread = Thread.currentThread();if ((thread instanceof ForkJoinWorkerThread) &&(wt = (ForkJoinWorkerThread)thread).pool == this) {// 如果当前线程是worker线程,则消费自己队列中的任务helpQuiescePool(wt.workQueue);return true;}// 非pool中的内部线程时,就尝试切分任务,即stealing任务long startTime = System.nanoTime();WorkQueue[] ws;int r = 0, m;boolean found = true;while (!isQuiescent() && (ws = workQueues) != null &&(m = ws.length - 1) >= 0) {if (!found) {if ((System.nanoTime() - startTime) > nanos)return false;// 这里是重点,当没有可用资源时,申请释放占用的CPU时间片,// 避免一直循环判断,浪费宝贵的时间片Thread.yield(); // cannot block}found = false;for (int j = (m + 1) << 2; j >= 0; --j) {ForkJoinTask<?> t; WorkQueue q; int b, k;if ((k = r++ & m) <= m && k >= 0 && (q = ws[k]) != null &&(b = q.base) - q.top < 0) {found = true;// 执行子任务if ((t = q.pollAt(b)) != null)t.doExec();break;}}}return true;}
}

ForkJoinPool的工作原理

参考文档:
java线程池(四):ForkJoinPool的使用及基本原理
Java中使用ForkJoinPool的实现示例

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

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

相关文章

Spring框架中哪些地方使用了反射

Spring框架中哪些地方使用了反射&#xff1f; 1. 依赖注入&#xff1a;Spring 使用反射机制获取对象并进行属性注入&#xff0c;从而实现依赖注入。 2. AOP&#xff1a;Spring AOP 使用 JDK 动态代理或者 CGLIB 字节码增强技术来实现 AOP 的切面逻辑&#xff0c;这其中就包含…

【无需任何插件】将VOS录音文件REC转MP3,REC转WAV或MP3的具体流程

【无需任何插件】将VOS录音文件REC转MP3&#xff0c;REC转WAV或MP3的具体流程&#xff0c;全过程完全复制本文命令即可&#xff0c;无需其他任何操作&#xff0c;命令长期有效&#xff0c;如失效&#xff0c;可以随时私信或者留言&#xff0c;免费更新。 __ 一位热爱鼓捣的IT爱…

20240613解决飞凌的OK3588-C的核心板的USB3.0接口不读U盘的问题

20240613解决飞凌的OK3588-C的核心板的USB3.0接口不读U盘的问题 2024/6/13 15:21 缘起&#xff0c;由于USB3.0的CC芯片在飞凌的OK3588-C的开发板的底板上&#xff0c;一切正常。 如果你单独使用核心板&#xff0c;很容易出现这个问题【省成本没有使用CC芯片】&#xff1a;不读U…

300PLC连接Modbus转Profibus网关与阀岛modbusRTU通讯

一、概况 300PLC作为常见的控制器设备&#xff0c;在与阀岛Modbus RTU通讯时&#xff0c;通常需要借助Modbus转Profibus网关&#xff08;XD-MDPB100&#xff09;来实现连接和数据交换。PLC通过Modbus转Profibus网关&#xff08;XD-MDPB100&#xff09;与阀岛Modbus RTU通讯是比…

labelme使用笔记:目标检测数据集标注和语义分割数据集批量生成

AI应用开发相关目录 本专栏包括AI应用开发相关内容分享&#xff0c;包括不限于AI算法部署实施细节、AI应用后端分析服务相关概念及开发技巧、AI应用后端应用服务相关概念及开发技巧、AI应用前端实现路径及开发技巧 适用于具备一定算法及Python使用基础的人群 AI应用开发流程概…

[AIGC] 使用Google的Guava库中的Lists工具类:常见用法详解

在Java程序设计中&#xff0c;集合是我们最常用的数据结构之一。为了方便我们操作集合&#xff0c;Google的Guava库提供了一个名为Lists的工具类&#xff0c;它封装了许多用于操作List对象的实用方法。在本文中&#xff0c;我们将详细介绍其常见的用法&#xff0c;以帮助您更好…

PyQt5 生成py文件不能运行;pushButton点击事件;QTextEdit 获取输入框内容

目录 cant open file c.pyuic: c.pyuic $FileName$ -o $FileNameWithoutExtension$.p PyQt5 生成py文件不能运行 pushButton点击事件 QTextEdit 获取输入框内容 整体运行代码: Creating a Qt Widget Based Application | Qt Creator Manual cant open file c.pyuic: c.…

NAT Easyip实验

我们这篇博客将重点讲述easy ip的配置&#xff1a; 以下面的一个简单的实验拓扑图为例&#xff1a; 本实验使用的网络地址&#xff1a; 1. 我们先来完成基础配置&#xff1a; 1.1AR1的基础配置&#xff1a; 1.2AR2上的基础配置 1.3完成AR1和AR2的基础配置后&#xff0c;我们…

数据库 | CSCI235/CSCI835 Database Systems Assignment 2

这个作业是完成与数据库事务的实现有关的任务 CSCI235/CSCI835 Database Systems Assignment 2 27 April 2020 Scope This assignment includes the tasks related to implementation of database transactions. The outcomes of the laboratory work are due by Saturday 16 M…

rust数据类型

目录 一&#xff0c;基本类型 1&#xff0c;基本类型 &#xff08;1&#xff09;整数类型 &#xff08;2&#xff09;浮点数 &#xff08;3&#xff09;bool类型 &#xff08;4&#xff09;char类型 2&#xff0c;基本类型的代数结构 二&#xff0c;复合类型 1&#x…

【信息学奥赛】CSP-J/S初赛04 进制转换相关问题(二、八、十六进制与十进制互相转换)

​ 大家好&#xff0c;我是bigbigli&#xff0c;今天我们学习初赛中常考的一个数学内容——进制转换 进制转换 常用的数制 权&#xff1a;数制中某一位上的1所表示的数值的大小&#xff0c;比如十进制中的123&#xff0c;1的位权是100,2的位权是10,3的位权是1。 进制基数基数…

这些帮助你成长的IOS应用,建议收藏

TrackIt TrackIt是一款功能丰富的任务清单、日程管理和习惯打卡应用&#xff0c;旨在帮助用户提高效率和专注力。通过这些功能&#xff0c;用户可以更好地规划时间和任务&#xff0c;从而实现个人目标和养成良好习惯。 在目标设定方面&#xff0c;SMART原则是一个常用的方法&a…

【决战欧洲之巅】丹麦 vs 英格兰战术分析和球员状态以及走地数据分析

丹麦 vs 英格兰战术分析和球员状态 阵型与战术 丹麦: 阵型:丹麦通常采用4-3-3阵型&#xff0c;以因防守为基础&#xff0c;同时通过快速反击等找机会。 ”战术: 防守:丹麦在防守时以紧密的四后卫体彩为核心&#xff0c;辅以中场球员的积极回防&#xff0c;形成稳因的防 线。…

解决HassOS无法获取ip地址问题

目录 问题描述解法 问题描述 在ESXi中安装完HassOS后&#xff0c;开机遇到一个无法获取ip地址的问题&#xff0c;如下图所示&#xff1a; 输入network info命令&#xff0c;显示ipv4已被禁用&#xff0c;如下图所示&#xff1a; 解法 在控制台ha >后输入下面命令 netw…

基于Java技术“漫画之家”系统

开头语&#xff1a;你好呀&#xff0c;我是计算机学姐码农小野&#xff01;如果有相关需求&#xff0c;可以私信联系我。 开发语言&#xff1a;Java 数据库&#xff1a;MySQL 技术&#xff1a;Java技术&#xff0c;B/S结构&#xff0c;SpringBoot框架 工具&#xff1a;MyEc…

使用Python写入Excel文件,设置列宽和行高

哈喽,大家好,我是木头左! Python与Excel的完美结合 在当今这个数据爆炸的时代,处理大量数据已经成为了日常工作的一部分。而Python作为一种功能强大、易于学习的编程语言,已经成为了数据处理的首选工具。那么,如何将Python与Excel结合起来,实现数据的高效处理呢?本文将…

DBdoctor v3.2.1 版本发布,支持对瀚高、人大金仓的纳管!

DBdoctor v3.2.1版本发布&#xff0c;更新内容如下&#xff1a; 功能优化 SQL审核&#xff1a;新增批量SQL审核功能&#xff0c;支持通过文件上传或者输入窗口批量提交的方式进行SQL审核。 新增数据库引擎 新增对瀚高、人大金仓数据库引擎的纳管支持。 瀚高&#xff08;highg…

板凳--------第60章 SOCKET:服务器设计

60.1 迭代型和并发型服务器 1016 1.迭代型&#xff1a; 服务器每次只处理一个客户端&#xff0c;只有当完全处理完一个客户端的请求后才会去处理下一个客户端。只适用于快速处理客户端请求的场景&#xff0c;因为每个客户端都必须等待&#xff0c;直到前面所有的客户端都处理完…

一键批量复制至指定文件夹,轻松实现同名文件覆盖,数据管理更高效!

在数字化时代&#xff0c;文件的管理与复制是每位电脑用户都不可避免的日常工作。你是否曾因为大量文件的复制与更新而焦头烂额&#xff1f;是否曾因为同名文件的冲突而不知所措&#xff1f;别担心&#xff0c;现在&#xff0c;我们为您带来高效文件复制管理的秘诀&#xff0c;…

【管理咨询宝藏132】国际顶级咨询公司战略组织运营报告套装

本报告首发于公号“管理咨询宝藏”&#xff0c;如需阅读完整版报告内容&#xff0c;请查阅公号“管理咨询宝藏”。 【管理咨询宝藏132】国际顶级咨询公司战略&组织&运营报告套装 【格式】PDF版本 【关键词】德勤、罗兰贝格、外资咨询、战略规划、组织管控、运营提升、…