Thread.onSpinWait()有什么作用?为什么要睡眠0毫秒?

2024-03-19 23:20

本文主要是介绍Thread.onSpinWait()有什么作用?为什么要睡眠0毫秒?,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

概述

今天在整理之前学习资料时,偶然看见之前自己写的demo:

public class MyTest {static volatile boolean temp = true;public static void main(String[] args) throws InterruptedException {Thread thread = new Thread(() -> {while (temp) {Thread.onSpinWait(); // Thread.sleep(0);}System.out.print("检测到变量为false,退出循环");});thread.start();Thread.sleep(3000L);temp = false;}}

运行结果:

检测到变量为false,退出循环

为了使线程能够更快的循环,以便让我能够及时的知道temp的状态,尽快的进行下一次循环,在方法中我比较粗暴的加入了Thread.onSpinWait()方法,Thread.onSpinWait()方法大家可以认为是Thread.sleep(0)的作用,

那么我为什么要加一个睡眠0毫秒的动作呢?让线程挂起0毫秒有什么用途呢?

线程状态

在Java中,线程有三个基本的状态:就绪状态(Runnable)、运行状态(Running)和阻塞状态(Blocked)。

  1. 就绪状态(Runnable):当线程被创建并启动后,它进入就绪状态。在就绪状态下,线程已经准备好执行,但还没有获取到CPU的执行时间片。线程处于就绪状态时,可以被调度器选择为下一个要执行的线程。
  2. 运行状态(Running):当线程获取到CPU的执行时间片时,它进入运行状态。在运行状态下,线程正在执行其任务代码。线程会一直保持运行状态,直到它主动放弃CPU的执行时间片,或者被其他高优先级线程抢占CPU。
  3. 阻塞状态(Blocked):线程在某些情况下会进入阻塞状态。当线程在执行过程中遇到某些阻塞的情况,比如等待I/O操作、等待获取锁、等待其他线程的通知等,它会进入阻塞状态。在阻塞状态下,线程暂时停止执行,不会占用CPU资源。当阻塞条件满足时,线程会被唤醒并重新进入就绪状态,等待获取CPU执行时间片。

线程的状态转换如下:

  • 就绪状态 -> 运行状态:当线程被调度器选择为下一个要执行的线程时,它从就绪状态转换为运行状态。
  • 运行状态 -> 就绪状态:线程主动调用yield()方法或者sleep()方法,或者被其他高优先级线程抢占CPU时,它从运行状态转换为就绪状态。
  • 运行状态 -> 阻塞状态:线程在执行过程中遇到阻塞条件,比如等待I/O操作或获取锁时,它从运行状态转换为阻塞状态。
  • 阻塞状态 -> 就绪状态:当阻塞条件满足时,线程被唤醒,从阻塞状态转换为就绪状态,等待获取CPU执行时间片。

线程的状态转换是由操作系统的调度器和Java虚拟机共同管理的。通过合理地管理线程的状态,可以实现多线程的并发执行和协作操作。

Thread.sleep(0)的意义

Java中,使用Thread.sleep(0)的目的是让当前线程主动放弃CPU的执行时间片,以便给其他具有相同优先级的线程执行的机会。虽然参数为0,但实际上并不是让线程休眠0毫秒,而是让线程进入就绪状态,等待重新获取CPU执行时间。

使用Thread.sleep(0)的主要意义在于提高多线程程序的公平性和响应性。当一个线程执行Thread.sleep(0)时,操作系统会重新调度其他就绪状态的线程,这样可以避免某个线程长时间占用CPU而导致其他线程无法得到执行的情况,从而提高了程序的公平性。

此外,Thread.sleep(0)还可以用于线程间的协作。当一个线程需要通知其他线程进行某些操作时,可以使用Thread.sleep(0)来主动放弃CPU执行时间,让其他线程有机会执行相应的操作。

Thread.onSpinWait()

@IntrinsicCandidate
public static void onSpinWait() {}

onSpinWait()方法是空实现,被@IntrinsicCandidate修饰,在JDK中,被@IntrinsicCandidate修饰的方法作为内部候选方法(intrinsic candidate)。内部候选方法是指可以由编译器或虚拟机进行特殊处理的方法,以提供更高效的执行方式或更好的性能。

简单来说就是jdk对Thread.onSpinWait()方法进行了特殊优化,那么优化后的效率到底有没有提升呢?

public class MyTest {public static void main(String[] args) throws InterruptedException {long start = System.currentTimeMillis();for (int i = 0; i < 100000000; i++) {Thread.sleep(0);}System.out.println(System.currentTimeMillis() - start);start = System.currentTimeMillis();for (int i = 0; i < 100000000; i++) {Thread.onSpinWait();}System.out.println(System.currentTimeMillis() - start);start = System.currentTimeMillis();for (int i = 0; i < 100000000; i++) {}System.out.println(System.currentTimeMillis() - start);}
}

运行结果

23224
2
0

上述程序,循环一亿次可以看出,在速度方面 空循环 > Thread.onSpinWait() > Thread.sleep(0), 空循环和Thread.onSpinWait()仅存在细微差别

在cpu利用方面: Thread.onSpinWait() = Thread.sleep(0) > 空循环

这篇关于Thread.onSpinWait()有什么作用?为什么要睡眠0毫秒?的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Spring Boot项目部署命令java -jar的各种参数及作用详解

《SpringBoot项目部署命令java-jar的各种参数及作用详解》:本文主要介绍SpringBoot项目部署命令java-jar的各种参数及作用的相关资料,包括设置内存大小、垃圾回收... 目录前言一、基础命令结构二、常见的 Java 命令参数1. 设置内存大小2. 配置垃圾回收器3. 配置线程栈大小

Java编译生成多个.class文件的原理和作用

《Java编译生成多个.class文件的原理和作用》作为一名经验丰富的开发者,在Java项目中执行编译后,可能会发现一个.java源文件有时会产生多个.class文件,从技术实现层面详细剖析这一现象... 目录一、内部类机制与.class文件生成成员内部类(常规内部类)局部内部类(方法内部类)匿名内部类二、

C++ 中的 if-constexpr语法和作用

《C++中的if-constexpr语法和作用》if-constexpr语法是C++17引入的新语法特性,也被称为常量if表达式或静态if(staticif),:本文主要介绍C++中的if-c... 目录1 if-constexpr 语法1.1 基本语法1.2 扩展说明1.2.1 条件表达式1.2.2 fa

css中的 vertical-align与line-height作用详解

《css中的vertical-align与line-height作用详解》:本文主要介绍了CSS中的`vertical-align`和`line-height`属性,包括它们的作用、适用元素、属性值、常见使用场景、常见问题及解决方案,详细内容请阅读本文,希望能对你有所帮助... 目录vertical-ali

浅析CSS 中z - index属性的作用及在什么情况下会失效

《浅析CSS中z-index属性的作用及在什么情况下会失效》z-index属性用于控制元素的堆叠顺序,值越大,元素越显示在上层,它需要元素具有定位属性(如relative、absolute、fi... 目录1. z-index 属性的作用2. z-index 失效的情况2.1 元素没有定位属性2.2 元素处

Spring 中 BeanFactoryPostProcessor 的作用和示例源码分析

《Spring中BeanFactoryPostProcessor的作用和示例源码分析》Spring的BeanFactoryPostProcessor是容器初始化的扩展接口,允许在Bean实例化前... 目录一、概览1. 核心定位2. 核心功能详解3. 关键特性二、Spring 内置的 BeanFactory

Spring组件初始化扩展点BeanPostProcessor的作用详解

《Spring组件初始化扩展点BeanPostProcessor的作用详解》本文通过实战案例和常见应用场景详细介绍了BeanPostProcessor的使用,并强调了其在Spring扩展中的重要性,感... 目录一、概述二、BeanPostProcessor的作用三、核心方法解析1、postProcessB

MyBatis的配置对象Configuration作用及说明

《MyBatis的配置对象Configuration作用及说明》MyBatis的Configuration对象是MyBatis的核心配置对象,它包含了MyBatis运行时所需的几乎所有配置信息,这个对... 目录MyBATis配置对象Configuration作用Configuration 对象的主要作用C

MySQL表锁、页面锁和行锁的作用及其优缺点对比分析

《MySQL表锁、页面锁和行锁的作用及其优缺点对比分析》MySQL中的表锁、页面锁和行锁各有特点,适用于不同的场景,表锁锁定整个表,适用于批量操作和MyISAM存储引擎,页面锁锁定数据页,适用于旧版本... 目录1. 表锁(Table Lock)2. 页面锁(Page Lock)3. 行锁(Row Lock

Mysql DATETIME 毫秒坑的解决

《MysqlDATETIME毫秒坑的解决》本文主要介绍了MysqlDATETIME毫秒坑的解决,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着... 今天写代码突发一个诡异的 bug,代码逻辑大概如下。1. 新增退款单记录boolean save = s