什么是JDK21虚拟线程

2024-05-26 02:12
文章标签 线程 虚拟 jdk21

本文主要是介绍什么是JDK21虚拟线程,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

JDK21虚拟线程

  • 1. 来一段小故事
  • 2. 什么是虚拟线程
  • 3. 虚拟线程的几个关键特点
  • 4.细说关键特点
    • 1.为什么轻量级的
      • 1.传统线程运行时间
      • 2.虚拟线程运行时间
      • 3.对垃圾回收的影响
    • 2.非绑定OS线程的魅力所在
    • 3.和传统相比为何易于使用
    • 4.阻塞优化有什么好处
      • 1.什么是阻塞优化
      • 2.JDK 21虚拟线程的阻塞优化
      • 3.传统线程的阻塞

1. 来一段小故事

  1. 假设博主经营一家快递公司,以前呢,每送一件包裹,你都得安排一辆大卡车出去,哪怕包裹很小。这样操作虽然可靠,但是成本高,效率低,特别是当有很多小包裹要送的时候,大卡车们忙着到处跑,油费不少花,还经常堵在路上。
  2. JDK21的虚拟线程就像是引入了一种新型的送快递方式。现在,你可以用很多轻便的电动车来送包裹,这些电动车就是“虚拟线程”。它们成本低,启动快,数量可以很多,应对大量小任务轻轻松松。当电动车(虚拟线程)在等红灯或者充电(执行耗时操作如读写文件)时,司机(JVM)就会让其他电动车接手其他包裹,保证路上总有车在跑,效率大大提升。
  3. 而且,用这些电动车安排送货任务非常简单,就像以前安排卡车一样,只是现在你有了更灵活、更高效的工具。当然,对于那些确实需要大卡车的大件货物(重量级计算任务),你还是可以用传统的卡车(操作系统线程),两种方式结合使用,让快递业务更加高效顺畅。这就是JDK21虚拟线程的通俗解释。

2. 什么是虚拟线程

  1. 首先,让我们揭开虚拟线程的神秘面纱。虚拟线程,或称为协程,是一种由JVM直接管理的轻量级线程。不同于操作系统级别的传统线程,每个虚拟线程占用的资源极小,使得在同一进程中可以轻松创建成千上万条这样的线程,极大地提升了系统对于高并发场景的应对能力。

  2. Thread.ofVirtual():这是手动启动虚拟线程的简便方式,只需一行代码,你就能为特定任务分配一个虚拟线程。

  3. 结构化并发:JDK 21引入的预览特性之一,让并发控制更加有序和安全。通过结构化并发,程序可以在明确的生命周期边界内自动创建和管理虚拟线程,减少了死锁和竞态条件的风险。

  4. Executors的革新:类似线程池的使用模式,但针对虚拟线程进行了优化,让你能够以熟悉的API享受虚拟线程带来的性能提升。

  5. 用一串代码来体验一下:

public class VirtualThreadDemo {public static void main(String[] args) {// 创建一个虚拟线程工厂var threadFactory = Thread.ofVirtual().factory();// 使用虚拟线程执行任务for (int i = 0; i < 10_000; i++) {var vt = threadFactory.newThread(() -> {System.out.println('Hello from Virtual Thread: ' + Thread.currentThread());});vt.start();}// 等待所有虚拟线程完成(实际应用中需考虑更优雅的同步机制)// 这里仅作演示,未加入等待逻辑}
}
  1. 在上面代码中,我们使用Thread.ofVirtual().factory()创建了一个虚拟线程工厂,随后启动了1万个虚拟线程,每个线程打印出自己的信息。这在传统线程模型下几乎是不可想象的任务量,但在虚拟线程的支持下,却变得轻而易举

3. 虚拟线程的几个关键特点

  • 轻量级:虚拟线程的创建和销毁成本远低于操作系统线程,使得应用程序能够创建成千上万甚至百万级别的线程,这对于高并发场景特别有利。

  • 非绑定OS线程:虚拟线程不由操作系统直接管理,而是由Java虚拟机(JVM)管理。这意味着虚拟线程可以在较少的操作系统线程上实现复用,减少上下文切换开销和资源消耗。

  • 易于使用:开发者可以像创建常规线程一样创建虚拟线程,但不需要担心线程池大小调整或过多线程带来的性能问题。

  • 阻塞优化:当虚拟线程执行阻塞操作(如I/O操作、锁等待等)时,它们会被暂停,而其底层的载体线程(carrier thread,对应的操作系统线程)则可以被释放去执行其他虚拟线程,从而提高了整体的并发效率。

  • 调度由JVM控制:虚拟线程的生命周期、状态管理、任务提交、休眠和唤醒等操作完全由JVM控制,提供了更好的可控制性和灵活性。

4.细说关键特点

1.为什么轻量级的

1.传统线程运行时间

1.传统线程创建示例:

public class PlatformThreadExample {public static void main(String[] args) {long startTime = System.nanoTime();for (int i = 0; i < 10000; i++) {new Thread(() -> doWork()).start();}System.out.printf("创建==> %d 个线程,用时==> %d 纳秒",10000, System.nanoTime() - startTime);}private static void doWork() {// 简单的工作逻辑
}

2.运行结果
在这里插入图片描述
3.运行时间为:1041478300纳秒

2.虚拟线程运行时间

1.虚拟线程创建示例

import java.util.concurrent.ThreadFactory;public class VirtualThreadExample {public static void main(String[] args) {long startTime = System.nanoTime();ThreadFactory virtualThreadFactory = Thread.ofVirtual().factory();for (int i = 0; i < 1_000_000; i++) {Thread vt = virtualThreadFactory.newThread(() -> doWork());vt.start();}System.out.printf("创建==> %d 个线程,用时==> %d 秒",1_000_000, System.nanoTime() - startTime);}private static void doWork() {// 简单的工作逻辑}
}

2.运行结果
在这里插入图片描述
3.运行时间为:536852800纳秒

3.对垃圾回收的影响

  1. 资源消耗减少:虚拟线程相较于操作系统线程消耗更少的内存资源。因为它们不需要分配大量的栈空间(通常虚拟线程的栈空间可以动态调整且较小),减少了堆外内存的占用,这可能导致GC活动减少,尤其是在大量线程并发的场景下。

  2. 栈内存管理:虚拟线程的栈是动态分配和释放的,这意味着当虚拟线程不再使用或阻塞时,其占用的栈内存可以更快地被回收或复用,减少了长时间运行过程中累积的内存碎片,有助于GC更高效地进行内存整理。

  3. 生命周期管理:虚拟线程的生命周期通常较短,尤其是在处理短暂任务后迅速结束,这减少了需要跟踪和回收的对象数量,减轻了GC的压力。

  4. GC频率:在高并发场景下,由于每个虚拟线程的内存占用减少,整体的内存分配速率可能降低,导致GC事件的发生频率相对降低。

  5. GC停顿时间:由于虚拟线程的轻量级特性,它们对堆内存的影响减小,可能减少因大对象分配或老年代回收而导致的长停顿时间。

  6. 内存使用效率:虚拟线程栈的高效管理有助于维持稳定的内存使用水平,减少内存碎片,使得内存使用更加平滑,GC曲线可能展现出更加平稳的趋势。

2.非绑定OS线程的魅力所在

  1. 资源效率:虚拟线程消耗的内存远低于传统OS线程,因为它们共享JVM的资源,减少了对系统资源的争抢。

  2. 上下文切换成本低:JVM优化了虚拟线程之间的切换过程,几乎感受不到额外开销,提升了整体性能。

  3. 简化编程模型:开发者不再需要复杂的线程池配置,可以像处理普通对象一样创建和销毁虚拟线程,降低了并发编程的门槛。

  4. 用一个生活中的案例比喻:设想一家在线零售平台在大促期间面临亿级用户请求的挑战。使用虚拟线程前,服务器可能因线程管理和资源分配问题而崩溃。但在采用JDK 21后,每个用户请求都能被迅速封装进一个轻量级的虚拟线程中,JVM智能调度确保所有请求得到高效、有序的处理,不仅提升了用户体验,还显著降低了运维成本。

  5. 总而言之,JDK 21中的虚拟线程及其非绑定OS线程特性,它以极简的资源占用、高效的执行效率以及友好的编程模型,为开发者铺设了一条通往高性能并发应用的康庄大道。

3.和传统相比为何易于使用

1.先来用代码写一个示例:

public class HelloWorld {public static void main(String[] args) {Thread vThread = Thread.startVirtualThread(() -> {System.out.println('Hello, Virtual World!');});vThread.join(); // 等待虚拟线程结束}
}

2.就像代码中所写,创建一个虚拟线程就像调用:Thread.startVirtualThread(Runnable task)

3.这么简单,无需复杂的线程池配置,也不必担心过多线程导致的性能瓶颈

4.资源效率提升:传统线程每个都映射到操作系统层面,消耗显著资源。而虚拟线程则不然,它们数量众多却几乎不增加额外开销,使得应用程序能够更加灵活地应对高并发场景

4.阻塞优化有什么好处

1.什么是阻塞优化

1.用一个生活案例进行举例:假设博主正站在繁忙的十字路口,车辆川流不息,但偶尔因红灯而停滞不前,造成交通短暂拥堵。这就像我们的程序在执行过程中,线程遇到IO操作或锁竞争时被迫等待的情景。现在,想象有一种魔法,能让停滞的车辆瞬间消失,道路重新畅通无阻,直到绿灯亮起它们才再次出现——这就是JDK 21虚拟线程阻塞优化带给我们的奇迹。

2.阻塞优化的魅力:当虚拟线程遇到IO阻塞或类似情况时,JVM会施展它的“隐形斗篷”,将这个虚拟线程从其载体的平台线程上移除,释放该平台线程去处理其他任务。这一过程无需程序员显式编码,完全由JVM自动完成。相比之下,传统线程在阻塞时会占用一个操作系统线程,即使不做任何工作也是如此,白白浪费了宝贵的系统资源。

3.简要浏览一段代码:

public class VirtualThreadDemo {public static void main(String[] args) {// 创建一个虚拟线程执行网络请求Thread vThread = Thread.startVirtualThread(() -> {var response = fetchFromNetwork('https://editor.csdn.net/md?not_checkout=1&spm=1001.2014.3001.5352&articleId=139201961');System.out.println('Data fetched: ' + response);});// 主线程继续执行其他任务System.out.println('Main thread doing other work...');}static String fetchFromNetwork(String url) {// 假设这是一个耗时的网络请求// 在此期间,虚拟线程会被透明卸载,不会阻塞主线程或其他任务return 'dummy data';}
}

4.在这段代码中,当我们发起网络请求时,虚拟线程会自动处理潜在的阻塞,确保主线程和其他任务不受影响,展现了其高效的并发能力。

2.JDK 21虚拟线程的阻塞优化

  • 自动的非阻塞转换:虚拟线程在执行到阻塞操作时,JVM会自动将其从当前的载体线程(即实际的平台线程)上移除,释放载体线程去执行其他任务,而不会直接阻塞操作系统线程。这意味着即使虚拟线程阻塞,也不再消耗宝贵的系统资源。

  • 轻量级上下文切换:虚拟线程之间的上下文切换比传统线程更为轻量,因为它们不涉及操作系统级别的状态保存和恢复,减少了开销。

  • 透明性:对于开发者而言,虚拟线程上的阻塞操作看起来像是同步的,但底层实际上是以非阻塞方式高效处理,无需手动编写复杂的异步回调逻辑,代码更加简洁、直观。

  • 资源效率:由于虚拟线程不直接占用操作系统资源,可以创建数以百万计的线程而不会耗尽系统资源,使得高度并发的应用成为可能。

代码示例:


// 假设代码在JDK 21环境下,使用虚拟线程执行阻塞操作
import java.util.concurrent.*;public class VirtualThreadBlockingOptimized {public static void main(String[] args) {var executor = Executors.newVirtualThreadPerTaskExecutor();Future<?> future = executor.submit(() -> {try {// 同样是阻塞操作,但虚拟线程优化了阻塞处理Thread.sleep(1000);System.out.println("虚拟线程完成阻塞操作");} catch (InterruptedException e) {Thread.currentThread().interrupt();}});System.out.println("主线程继续执行,虚拟线程阻塞不会阻塞载体线程");try {// 等待虚拟线程完成,非必须,仅为演示future.get(); } catch (InterruptedException | ExecutionException e) {e.printStackTrace();}executor.shutdown();}
}

3.传统线程的阻塞

在传统的线程模型中,每个线程直接映射到操作系统的一个线程。当线程执行到阻塞操作,如I/O操作或等待锁时,操作系统会将该线程挂起,直到阻塞条件解除。传统线程的阻塞优化通常涉及:

  • 非阻塞I/O(NIO):使用如Java NIO来避免在I/O操作时阻塞线程,转而使用回调或者轮询机制来通知数据准备好。
  • 锁优化:如自旋锁、锁粗化、锁消除等技术减少线程因竞争锁而阻塞的情况。
  • 线程池:通过复用线程来减少频繁创建和销毁线程的开销,同时限制并发线程的数量以防止资源耗尽。

代码示例:

public class TraditionalThreadBlocking {public static void main(String[] args) throws InterruptedException {Thread thread = new Thread(() -> {try {// 阻塞操作,如读取文件或网络I/OThread.sleep(1000); System.out.println("传统线程完成阻塞操作");} catch (InterruptedException e) {Thread.currentThread().interrupt();}});thread.start();System.out.println("主线程继续执行,但系统资源被阻塞的线程占用");}
}

这篇关于什么是JDK21虚拟线程的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Java中实现线程的创建和启动的方法

《Java中实现线程的创建和启动的方法》在Java中,实现线程的创建和启动是两个不同但紧密相关的概念,理解为什么要启动线程(调用start()方法)而非直接调用run()方法,是掌握多线程编程的关键,... 目录1. 线程的生命周期2. start() vs run() 的本质区别3. 为什么必须通过 st

Linux实现线程同步的多种方式汇总

《Linux实现线程同步的多种方式汇总》本文详细介绍了Linux下线程同步的多种方法,包括互斥锁、自旋锁、信号量以及它们的使用示例,通过这些同步机制,可以解决线程安全问题,防止资源竞争导致的错误,示例... 目录什么是线程同步?一、互斥锁(单人洗手间规则)适用场景:特点:二、条件变量(咖啡厅取餐系统)工作流

Java中常见队列举例详解(非线程安全)

《Java中常见队列举例详解(非线程安全)》队列用于模拟队列这种数据结构,队列通常是指先进先出的容器,:本文主要介绍Java中常见队列(非线程安全)的相关资料,文中通过代码介绍的非常详细,需要的朋... 目录一.队列定义 二.常见接口 三.常见实现类3.1 ArrayDeque3.1.1 实现原理3.1.2

SpringBoot3中使用虚拟线程的完整步骤

《SpringBoot3中使用虚拟线程的完整步骤》在SpringBoot3中使用Java21+的虚拟线程(VirtualThreads)可以显著提升I/O密集型应用的并发能力,这篇文章为大家介绍了详细... 目录1. 环境准备2. 配置虚拟线程方式一:全局启用虚拟线程(Tomcat/Jetty)方式二:异步

如何解决Druid线程池Cause:java.sql.SQLRecoverableException:IO错误:Socket read timed out的问题

《如何解决Druid线程池Cause:java.sql.SQLRecoverableException:IO错误:Socketreadtimedout的问题》:本文主要介绍解决Druid线程... 目录异常信息触发场景找到版本发布更新的说明从版本更新信息可以看到该默认逻辑已经去除总结异常信息触发场景复

Linux网络配置之网桥和虚拟网络的配置指南

《Linux网络配置之网桥和虚拟网络的配置指南》这篇文章主要为大家详细介绍了Linux中配置网桥和虚拟网络的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 一、网桥的配置在linux系统中配置一个新的网桥主要涉及以下几个步骤:1.为yum仓库做准备,安装组件epel-re

JDK9到JDK21中值得掌握的29个实用特性分享

《JDK9到JDK21中值得掌握的29个实用特性分享》Java的演进节奏从JDK9开始显著加快,每半年一个新版本的发布节奏为Java带来了大量的新特性,本文整理了29个JDK9到JDK21中值得掌握的... 目录JDK 9 模块化与API增强1. 集合工厂方法:一行代码创建不可变集合2. 私有接口方法:接口

JAVA保证HashMap线程安全的几种方式

《JAVA保证HashMap线程安全的几种方式》HashMap是线程不安全的,这意味着如果多个线程并发地访问和修改同一个HashMap实例,可能会导致数据不一致和其他线程安全问题,本文主要介绍了JAV... 目录1. 使用 Collections.synchronizedMap2. 使用 Concurren

使用Python实现全能手机虚拟键盘的示例代码

《使用Python实现全能手机虚拟键盘的示例代码》在数字化办公时代,你是否遇到过这样的场景:会议室投影电脑突然键盘失灵、躺在沙发上想远程控制书房电脑、或者需要给长辈远程协助操作?今天我要分享的Pyth... 目录一、项目概述:不止于键盘的远程控制方案1.1 创新价值1.2 技术栈全景二、需求实现步骤一、需求

Spring Boot3虚拟线程的使用步骤详解

《SpringBoot3虚拟线程的使用步骤详解》虚拟线程是Java19中引入的一个新特性,旨在通过简化线程管理来提升应用程序的并发性能,:本文主要介绍SpringBoot3虚拟线程的使用步骤,... 目录问题根源分析解决方案验证验证实验实验1:未启用keep-alive实验2:启用keep-alive扩展建