JDK21|史诗级的更新,虚拟线程

2024-03-29 23:20

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

作者:鱼仔

博客首页: https://codeease.top

公众号:Java鱼仔

前言

要想看官方对于JDK21的更新说明,可以直接跳转到下面这个官方网站中

官网地址为:https://openjdk.org/projects/jdk/21/

JDK21是最新的LTS版本,里面添加了不少新的特性,本文将介绍JEP444–虚拟线程

新特性产生的动机

如果问我觉得JDK21中最大的更新是什么,那么我一定会选择虚拟线程。在JDK21之前,Java中创建的线程和
操作系统的内核线程是一对一的,就如下图这样:

调用操作系统的内核线程成本是很高的,因此我们会用池化技术去最大化提升线程性价比。但即使如此,针对IO密集型的并发任务,CPU并不是限制性能的瓶颈,线程数量才是。当前的这种线程实现还是严重限制了程序的吞吐量,于是JDK设计出了虚拟线程。在JDK19的时候第一次作为预览功能提出,JDK20第二次孵化,终于在JDK21的时候作为正式功能推出。

上图是虚拟线程的模拟图,不是说一个真实线程上只有两个虚拟线程,实际上,成千上万个虚拟线程可能只是在一个真实线程中运行。

如何使用虚拟线程

下面是四种创建虚拟线程的方式,为了减少学习成本,虚拟线程在使用上和普通线程十分相似

第一种通过Thread ofVirtual方法,代码如下:

public class VirtualThreadTest1 {public static void main(String[] args) {Thread.ofVirtual().start(() -> {System.out.println("run in virtual thread");});}
}

在start方法中传入Runnable方法即可

第二种方法通过Thread的startVirtualThread方法开启一个虚拟线程,代码如下:

public class VirtualThreadTest2 {public static void main(String[] args) {Thread.startVirtualThread(() -> {System.out.println("run in virtual thread");});}
}

第三种方法是通过ThreadFactory线程工厂创建虚拟线程,首先创建出一个线程工厂对象,接着调用工厂的newThread和start方法即可,代码如下:

public class VirtualThreadTest3 {public static void main(String[] args) {ThreadFactory factory = Thread.ofVirtual().factory();factory.newThread(()->{System.out.println("run in virtual thread");}).start();}
}

第四种方式是通过Executors新增的一个方法newVirtualThreadPerTaskExecutor来创建虚拟线程,然后在submit方法中传入Runnable方法即可。

public class VirtualThreadTest4 {public static void main(String[] args) {ExecutorService executorService = Executors.newVirtualThreadPerTaskExecutor();executorService.submit(()->{System.out.println("run in virtual thread");});}
}

需要注意的是,和之前创建线程不同,虚拟线程无需池化,因为可能一万个虚拟线程,也只是在一个实际的线程中运行。

两种线程的对比

虚拟线程的优势是在IO密集型的任务中,我通过一个实际的例子比较两者之间的性能差异。

首先创建了一个200个线程的定长线程池,然后创建了一万个任务,每个任务执行需要0.5秒,计算执行完成所有任务需要的时间;接着将定长线程池修改为虚拟线程,同样的代码逻辑运行。

public class ThreadCompare {public static void main(String[] args) {long startTime = System.currentTimeMillis();ExecutorService executor =  Executors.newFixedThreadPool(200);// 执行虚拟线程就替换成下面的语句。//       ExecutorService executor = Executors.newVirtualThreadPerTaskExecutor();for (int i = 0; i < 10000; i++) {executor.submit(()->{try {Thread.sleep(500);} catch (InterruptedException e) {throw new RuntimeException(e);}});}executor.close();System.out.printf("线程池耗时:%dms",System.currentTimeMillis()-startTime);}
}

下面是运行结果:

使用实际线程耗时25240ms,而通过虚拟线程只耗时776ms,遥遥领先的优势。

创建一万个虚拟线程可能也就用了几个实际线程,但是创建一万个实际线程直接就OOM。

总结

虚拟线程真的很强大,但是对虚拟线程最大的制约在于JDK8永不为奴。自己的项目还能玩玩JDK21,要将公司的项目JDK升级上去想都不用想。

这篇关于JDK21|史诗级的更新,虚拟线程的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

poj3468(线段树成段更新模板题)

题意:包括两个操作:1、将[a.b]上的数字加上v;2、查询区间[a,b]上的和 下面的介绍是下解题思路: 首先介绍  lazy-tag思想:用一个变量记录每一个线段树节点的变化值,当这部分线段的一致性被破坏我们就将这个变化值传递给子区间,大大增加了线段树的效率。 比如现在需要对[a,b]区间值进行加c操作,那么就从根节点[1,n]开始调用update函数进行操作,如果刚好执行到一个子节点,

hdu1394(线段树点更新的应用)

题意:求一个序列经过一定的操作得到的序列的最小逆序数 这题会用到逆序数的一个性质,在0到n-1这些数字组成的乱序排列,将第一个数字A移到最后一位,得到的逆序数为res-a+(n-a-1) 知道上面的知识点后,可以用暴力来解 代码如下: #include<iostream>#include<algorithm>#include<cstring>#include<stack>#in

hdu1689(线段树成段更新)

两种操作:1、set区间[a,b]上数字为v;2、查询[ 1 , n ]上的sum 代码如下: #include<iostream>#include<algorithm>#include<cstring>#include<stack>#include<queue>#include<set>#include<map>#include<stdio.h>#include<stdl

hdu 1754 I Hate It(线段树,单点更新,区间最值)

题意是求一个线段中的最大数。 线段树的模板题,试用了一下交大的模板。效率有点略低。 代码: #include <stdio.h>#include <string.h>#define TREE_SIZE (1 << (20))//const int TREE_SIZE = 200000 + 10;int max(int a, int b){return a > b ? a :

AI行业应用(不定期更新)

ChatPDF 可以让你上传一个 PDF 文件,然后针对这个 PDF 进行小结和提问。你可以把各种各样你要研究的分析报告交给它,快速获取到想要知道的信息。https://www.chatpdf.com/

GIS图形库更新2024.8.4-9.9

更多精彩内容请访问 dt.sim3d.cn ,关注公众号【sky的数孪技术】,技术交流、源码下载请添加微信:digital_twin123 Cesium 本期发布了1.121 版本。重大新闻,Cesium被Bentley收购。 ✨ 功能和改进 默认启用 MSAA,采样 4 次。若要关闭 MSAA,则可以设置scene.msaaSamples = 1。但是通过比较,发现并没有多大改善。

JavaFX应用更新检测功能(在线自动更新方案)

JavaFX开发的桌面应用属于C端,一般来说需要版本检测和自动更新功能,这里记录一下一种版本检测和自动更新的方法。 1. 整体方案 JavaFX.应用版本检测、自动更新主要涉及一下步骤: 读取本地应用版本拉取远程版本并比较两个版本如果需要升级,那么拉取更新历史弹出升级控制窗口用户选择升级时,拉取升级包解压,重启应用用户选择忽略时,本地版本标志为忽略版本用户选择取消时,隐藏升级控制窗口 2.

记录每次更新到仓库 —— Git 学习笔记 10

记录每次更新到仓库 文章目录 文件的状态三个区域检查当前文件状态跟踪新文件取消跟踪(un-tracking)文件重新跟踪(re-tracking)文件暂存已修改文件忽略某些文件查看已暂存和未暂存的修改提交更新跳过暂存区删除文件移动文件参考资料 咱们接着很多天以前的 取得Git仓库 这篇文章继续说。 文件的状态 不管是通过哪种方法,现在我们已经有了一个仓库,并从这个仓

消除安卓SDK更新时的“https://dl-ssl.google.com refused”异常的方法

消除安卓SDK更新时的“https://dl-ssl.google.com refused”异常的方法   消除安卓SDK更新时的“https://dl-ssl.google.com refused”异常的方法 [转载]原地址:http://blog.csdn.net/x605940745/article/details/17911115 消除SDK更新时的“

线程的四种操作

所属专栏:Java学习        1. 线程的开启 start和run的区别: run:描述了线程要执行的任务,也可以称为线程的入口 start:调用系统函数,真正的在系统内核中创建线程(创建PCB,加入到链表中),此处的start会根据不同的系统,分别调用不同的api,创建好之后的线程,再单独去执行run(所以说,start的本质是调用系统api,系统的api