亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

JDK21中虛擬線程到底是什么以及用法總結(jié)(看完便知)

 更新時(shí)間:2023年12月09日 09:32:24   作者:愛讀源碼的大都督  
這篇文章主要給大家介紹了關(guān)于JDK21中虛擬線程到底是什么以及用法的相關(guān)資料,虛擬線程是一種輕量化的線程封裝,由jvm直接調(diào)度和管理,反之普通的線程其實(shí)是調(diào)用的操作系統(tǒng)的能力,對(duì)應(yīng)的是操作系統(tǒng)級(jí)的線程,需要的朋友可以參考下

本文涉及到的技術(shù):虛擬線程、結(jié)構(gòu)化并發(fā)、線程池、TheadLocal,對(duì)原理感興趣的可以直接跳到原理部分。

虛擬線程是JDK19中引入的,JDK21正式發(fā)布,我們先來(lái)看看虛擬線程的幾種用法,然后再來(lái)分析底層實(shí)現(xiàn)原理。

先定義一個(gè)Runnable:

通過(guò)觀察輸出結(jié)果,就能知道當(dāng)前運(yùn)行Task的是不是虛擬線程。

也可以增加以下代碼直接判斷是不是虛擬線程:

Thread.ofVirtual()

手動(dòng)開啟虛擬線程執(zhí)行任務(wù):

自動(dòng)開啟虛擬線程執(zhí)行任務(wù):

兩者輸出結(jié)果類似,為:

根據(jù)名字可以看出確實(shí)是用的VirtualThread,但似乎跟ForkJoinPool有關(guān),后面會(huì)分析。

我們也可以通過(guò)以下方式來(lái)創(chuàng)建普通線程

輸出結(jié)果為:

確實(shí)是普通線程。

還可以先得到一個(gè)ThreadFactory,然后來(lái)創(chuàng)建虛擬線程:

輸出結(jié)果為:

還有一種更簡(jiǎn)單的API:

輸出結(jié)果為:

結(jié)構(gòu)化并發(fā)

在JDK21中還有一個(gè)新特性(預(yù)覽版),叫做結(jié)構(gòu)化并發(fā),也會(huì)自動(dòng)創(chuàng)建虛擬線程來(lái)運(yùn)行代碼,比如:

輸出結(jié)果為:

Executors

還有一種和線程池類似的使用方式:

以上代碼中每個(gè)任務(wù)運(yùn)行時(shí)都會(huì)開啟一個(gè)虛擬線程,輸出結(jié)果為:

表示有3個(gè)虛擬線程。

虛擬線程底層原理

以上大概就是使用或創(chuàng)建虛擬線程的幾種情況了,那到底什么是虛擬線程呢?它跟線程有什么關(guān)系?它跟ForkJoinPool又有什么關(guān)系呢?

虛擬線程畢竟是虛擬的,就像虛擬機(jī)也是虛擬的,是需要真實(shí)操作系統(tǒng)來(lái)支撐運(yùn)行的。而虛擬線程仍然是基于線程來(lái)進(jìn)行調(diào)度執(zhí)行的。

我們先來(lái)看看普通線程的缺點(diǎn)在哪,看下面代碼:

假如是一個(gè)普通線程執(zhí)行上述代碼,在輸出完“before”后,線程就會(huì)睡眠1秒,然后才會(huì)輸出“after”,如果是一個(gè)線程要執(zhí)行3個(gè)這樣的任務(wù),比如:

生成一個(gè)只有一個(gè)線程的線程池,用它來(lái)執(zhí)行三個(gè)任務(wù),實(shí)際上就是串行執(zhí)行這三個(gè)任務(wù),輸出結(jié)果為:

但是,我們好好想想:當(dāng)這個(gè)普通線程執(zhí)行完第一個(gè)任務(wù)的“before”后,需要等1s才執(zhí)行“after”,那能不能在等1s的過(guò)程中去執(zhí)行第二個(gè)任務(wù)的“before”呢?原則上是可以的,這就是虛擬線程要優(yōu)化的點(diǎn)。

大家好好理解一下上面的這句話,這是精髓

我們來(lái)看改成虛擬線程后的運(yùn)行效果,先修改Task:

然后運(yùn)行:

輸出結(jié)果為:

大家運(yùn)行時(shí)可能會(huì)發(fā)現(xiàn)有多個(gè)不同的ForkJoinPool-1-worker,那是因?yàn)槲易隽伺渲?,后面?huì)解釋

不知道大家能不能看懂這個(gè)效果,我們可以發(fā)現(xiàn)有3個(gè)虛擬線程:VirtualThread[#21]、VirtualThread[#23]、VirtualThread[#24],但是只有一個(gè)線程:ForkJoinPool-1-worker-1,雖然只有一個(gè)線程,卻達(dá)到了并行執(zhí)行三個(gè)任務(wù)的效果,其原理就是上面所分析的:

線程先執(zhí)行任務(wù)1,任務(wù)1睡眠的過(guò)程中,線程會(huì)去執(zhí)行任務(wù)2任務(wù)2睡眠的過(guò)程中,線程會(huì)去執(zhí)行任務(wù)3任務(wù)3睡眠的過(guò)程中,線程暫時(shí)沒有任務(wù)執(zhí)行了過(guò)一會(huì),任務(wù)1睡眠結(jié)束,線程繼續(xù)執(zhí)行任務(wù)1然后,任務(wù)2睡眠結(jié)束,線程繼續(xù)執(zhí)行任務(wù)2最后,任務(wù)3睡眠結(jié)束,線程繼續(xù)執(zhí)行任務(wù)3

這樣就達(dá)到了一個(gè)線程并行執(zhí)行三個(gè)任務(wù)的效果,從中,我們可以看到,線程需要知道:一個(gè)任務(wù)什么時(shí)候開始睡眠了,什么時(shí)候睡眠結(jié)束了,哪個(gè)任務(wù)還沒開始執(zhí)行,哪個(gè)任務(wù)已經(jīng)在執(zhí)行中了?

但是,任務(wù)是程序員所定義的,所以就需要虛擬線程來(lái)封裝任務(wù),而線程只關(guān)心虛擬線程即可,也就是線程負(fù)責(zé)來(lái)調(diào)度各個(gè)虛擬線程的執(zhí)行,也就是來(lái)判斷虛擬線程是不是睡眠了?是不是正在運(yùn)行?

我們可以把虛擬線程理解為一個(gè)對(duì)象,這個(gè)虛擬線程對(duì)象有幾種狀態(tài),比如是不是睡眠中,是不是運(yùn)行中,而一個(gè)線程可以支持同時(shí)運(yùn)行多個(gè)虛擬線程對(duì)象,當(dāng)線程發(fā)現(xiàn)某個(gè)虛擬線程對(duì)象睡眠時(shí),就會(huì)去運(yùn)行其他的虛擬線程對(duì)象。

或者這么說(shuō):虛擬線程sleep了,底層的線程并不一定sleep了。

所以,虛擬線程就是線程調(diào)度的單位,一個(gè)線程可以調(diào)度很多個(gè)虛擬線程,如果有多個(gè)線程,那當(dāng)然就能調(diào)度更多虛擬線程了,所以在JDK的實(shí)現(xiàn)中,使用ForkJoinPool來(lái)提供線程作為虛擬線程的調(diào)度者,同時(shí)由于ForkJoinPool的任務(wù)竊取機(jī)制,能進(jìn)一步提高任務(wù)并行執(zhí)行的效率。

默認(rèn)情況下,這個(gè)ForkJoinPool中的線程數(shù)等于CPU核心數(shù),我們可以通過(guò)以下參數(shù)來(lái)修改:

  • -Djdk.virtualThreadScheduler.parallelism=1
  • -Djdk.virtualThreadScheduler.maxPoolSize=1

另外,虛擬線程也不用考慮池化,因?yàn)樗幌窬€程,開啟和關(guān)閉一個(gè)線程是需要調(diào)用操作系統(tǒng)的,而虛擬線程跟操作系統(tǒng)沒關(guān)系。

再另外,JDK21中的虛擬線程也支持ThreadLocal,也就是一個(gè)虛擬線程在執(zhí)行任務(wù)的過(guò)程中,也可以通過(guò)ThreadLocal來(lái)共享數(shù)據(jù),使得我們?cè)陂_發(fā)過(guò)程中就把虛擬線程當(dāng)作普通線程使用就可以了。

還有要注意的是,當(dāng)任務(wù)進(jìn)行網(wǎng)絡(luò)IO、磁盤IO時(shí)也相當(dāng)是sleep了,所以如果虛擬線程用到真實(shí)項(xiàng)目中,就能做到用少量線程支撐較高的并發(fā),從而能大大提高項(xiàng)目的吞吐量,虛擬線程不是用來(lái)提速的,而是用來(lái)提高吞吐量的

總結(jié)

到此這篇關(guān)于JDK21中虛擬線程到底的文章就介紹到這了,更多相關(guān)JDK21虛擬線程用法內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • mybatisPlus打印sql配置的四種方法

    mybatisPlus打印sql配置的四種方法

    MyBatis-Plus提供了多種方式來(lái)配置SQL打印功能,以便進(jìn)行調(diào)試和性能分析,這篇文章主要介紹了mybatisPlus打印sql配置的四種方法,文中通過(guò)代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2024-12-12
  • MyBatis屬性名和字段名配置不一致的解決

    MyBatis屬性名和字段名配置不一致的解決

    在使用MyBatis進(jìn)行持久化框架開發(fā)時(shí),經(jīng)常會(huì)遇到屬性名和數(shù)據(jù)庫(kù)字段名不一致的情況,本文主要介紹了MyBatis屬性名和字段名配置不一致的解決,具有一定的參考價(jià)值,感興趣的可以了解一下
    2024-06-06
  • Java動(dòng)態(tài)代理分析及簡(jiǎn)單實(shí)例

    Java動(dòng)態(tài)代理分析及簡(jiǎn)單實(shí)例

    這篇文章主要介紹了 Java動(dòng)態(tài)代理分析及簡(jiǎn)單實(shí)例的相關(guān)資料,需要的朋友可以參考下
    2017-02-02
  • IDEA中的.iml文件和.idea文件的作用及意義

    IDEA中的.iml文件和.idea文件的作用及意義

    iml是 項(xiàng)目標(biāo)識(shí)文件,每一個(gè)模塊都有一個(gè)iml文件,存儲(chǔ)模塊的相關(guān)信息,跟eclipse的project文件是一樣的功能,這篇文章主要介紹了IDEA中的.iml文件和.idea文件的作用,需要的朋友可以參考下
    2023-09-09
  • Java中CAS機(jī)制實(shí)現(xiàn)方法詳解

    Java中CAS機(jī)制實(shí)現(xiàn)方法詳解

    傳統(tǒng)的并發(fā)控制手段如synchronized和ReentrantLock雖有效防止資源競(jìng)爭(zhēng),卻可能引起性能開銷,相比之下,CAS(CompareAndSwap)提供一種輕量級(jí)的樂觀鎖策略,通過(guò)硬件級(jí)別的原子指令實(shí)現(xiàn)無(wú)鎖并發(fā),提高性能,需要的朋友可以參考下
    2024-09-09
  • Java數(shù)據(jù)結(jié)構(gòu)之LinkedList的用法詳解

    Java數(shù)據(jù)結(jié)構(gòu)之LinkedList的用法詳解

    鏈表(Linked?list)是一種常見的基礎(chǔ)數(shù)據(jù)結(jié)構(gòu),是一種線性表。Java的LinkedList(鏈表)?類似于?ArrayList,是一種常用的數(shù)據(jù)容器,本文就來(lái)簡(jiǎn)單講講它的使用吧
    2023-05-05
  • Jackson常用方法以及jacksonUtil工具類詳解

    Jackson常用方法以及jacksonUtil工具類詳解

    這篇文章主要介紹了Jackson常用方法以及jacksonUtil工具類詳解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-06-06
  • Java實(shí)現(xiàn)在線考試系統(tǒng)與設(shè)計(jì)(學(xué)生功能)

    Java實(shí)現(xiàn)在線考試系統(tǒng)與設(shè)計(jì)(學(xué)生功能)

    這篇文章主要介紹了Java實(shí)現(xiàn)在線考試系統(tǒng)與設(shè)計(jì)(學(xué)生功能),本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-02-02
  • 使用Java反射模擬實(shí)現(xiàn)Spring的IoC容器的操作

    使用Java反射模擬實(shí)現(xiàn)Spring的IoC容器的操作

    這篇文章主要介紹了使用Java反射模擬實(shí)現(xiàn)Spring的IoC容器的操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-08-08
  • 老生常談Scanner的基本用法

    老生常談Scanner的基本用法

    下面小編就為大家?guī)?lái)一篇老生常談Scanner的基本用法。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2017-07-07

最新評(píng)論