日韩成人免费在线_国产成人一二_精品国产免费人成电影在线观..._日本一区二区三区久久久久久久久不

當前位置:首頁 > 科技  > 軟件

Java 新技術:虛擬線程使用指南

來源: 責編: 時間:2024-01-15 09:21:07 266觀看
導讀虛擬線程是在 Java 21 版本中實現的一種輕量級線程。它由 JVM 進行創建以及管理。虛擬線程和傳統線程(我們稱之為平臺線程)之間的主要區別在于,我們可以輕松地在一個 Java 程序中運行大量、甚至數百萬個虛擬線程。本文是

虛擬線程是在 Java 21 版本中實現的一種輕量級線程。它由 JVM 進行創建以及管理。虛擬線程和傳統線程(我們稱之為平臺線程)之間的主要區別在于,我們可以輕松地在一個 Java 程序中運行大量、甚至數百萬個虛擬線程。fXt28資訊網——每日最新資訊28at.com

fXt28資訊網——每日最新資訊28at.com

本文是繼《Java 21 新技術:虛擬線程使用指南》的第二篇文章,無意全面涵蓋虛擬線程的每個重要細節,目的是給大家使用虛擬線程提供一套使用指南,幫助大家能更好使用的虛擬線程,發揮其作用并避免踩坑。fXt28資訊網——每日最新資訊28at.com

本文完整大綱如下,fXt28資訊網——每日最新資訊28at.com

fXt28資訊網——每日最新資訊28at.com

使用信號量限制并發

在某些場景下,我們需要限制某個操作的并發數。例如某些外部服務可能無法同時處理超過 10 個并發請求。fXt28資訊網——每日最新資訊28at.com

由于平臺線程是一種寶貴的資源,通常在線程池中進行管理,因此線程池的使用對于如今的程序員相當普遍。fXt28資訊網——每日最新資訊28at.com

fXt28資訊網——每日最新資訊28at.com

比如上面例子要限制并發請求數,某些人會使用線程池來處理,代碼如下,fXt28資訊網——每日最新資訊28at.com

ExecutorService es = Executors.newFixedThreadPool(10);...Result foo() {    try {        var fut = es.submit(() -> callLimitedService());        return f.get();    } catch (...) { ... }}

上面代碼示例可以確保外部服務最多只有 10 個并發請求,因為我們的線程池中只有最多 10 個線程。fXt28資訊網——每日最新資訊28at.com

限制并發只是使用線程池的副產品。線程池旨在共享稀缺資源,而虛擬線程并不稀缺,因此永遠不應該池化虛擬線程!fXt28資訊網——每日最新資訊28at.com

使用虛擬線程時,如果要限制訪問某些服務的并發請求,則應該使用專門為此目的設計的 Semaphore 類。示例代碼如下,fXt28資訊網——每日最新資訊28at.com

Semaphore sem = new Semaphore(10);...Result foo() {    sem.acquire();    try {        return callLimitedService();    } finally {        sem.release();    }}

在這個示例中,同一時刻只有 10 個虛擬線程可以進入 foo() 方法取得鎖,而其他虛擬線程將會被阻塞。fXt28資訊網——每日最新資訊28at.com

簡單地使用信號量阻塞某些虛擬線程可能看起來與將任務提交到固定數量線程池有很大不同,但事實并非如此。fXt28資訊網——每日最新資訊28at.com

將任務提交到等待任務池會將它們排隊處理,信號量在內部(或任何其他阻塞同步構造)構造了一個阻塞線程隊列,這些任務在阻塞線程隊列上也會進行排隊處理。fXt28資訊網——每日最新資訊28at.com

fXt28資訊網——每日最新資訊28at.com

我們可以將平臺線程池認作是從等待任務隊列中提取任務進行處理的工作人員,然后將虛擬線程視為任務本身,在任務或者線程可以執行之前將會被阻塞,但任務或者線程被阻塞時在計算機中的底層表示上實際是相同的。fXt28資訊網——每日最新資訊28at.com

這里想告訴大家的就是不管是線程池的任務排隊,還是信號量內部的線程阻塞,它們之間是由等效性的。在虛擬線程某些需要限制并發數場景下,直接使用信號量即可。fXt28資訊網——每日最新資訊28at.com

不要在線程局部變量中緩存可重用對象

虛擬線程支持線程局部變量,就像平臺線程一樣。通常線程局部變量用于將一些特定于上下文的信息與當前運行的代碼關聯起來,例如當前事務和用戶 ID。fXt28資訊網——每日最新資訊28at.com

對于虛擬線程來說,使用線程局部變量是完全合理的。但是如果考慮更安全、更有效的線程局部變量,可以使用 Scoped Values。fXt28資訊網——每日最新資訊28at.com

更多有關 Scoped Values 介紹,請參閱 https://docs.oracle.com/en/java/javase/21/core/scoped-values.html#GUID-9A4565C5-82AE-4F03-A476-3EAA9CDEB0F6fXt28資訊網——每日最新資訊28at.com

fXt28資訊網——每日最新資訊28at.com

線程局部變量有一種用途與虛擬線程是不太適合的,那就是緩存可重用對象。fXt28資訊網——每日最新資訊28at.com

可重用對象的創建成本通常很高,通常消耗大量內存且可變,還不是線程安全的。它們被緩存在線程局部變量中,以減少它們實例化的次數以及它們在內存中的實例數量,好處是它們可以被線程上不同時間運行的多個任務重用,減少昂貴對象創建的開銷。fXt28資訊網——每日最新資訊28at.com

例如 SimpleDateFormat 的實例創建成本很高,而且不是線程安全的。為了解決創建成本、線程不安全問題,通常是將此類實例緩存在 ThreadLocal 中,如下例所示:fXt28資訊網——每日最新資訊28at.com

static final ThreadLocal<SimpleDateFormat> cachedFormatter =       ThreadLocal.withInitial(SimpleDateFormat::new);void foo() {  ... cachedFormatter.get().format(...); ...}

僅當線程(以及因此在線程本地緩存的昂貴對象)被多個任務共享和重用時(就像平臺線程被池化時的情況一樣),這種緩存才有用。許多任務在線程池中運行時可能會調用 foo,但由于池中僅包含幾個線程,因此該對象只會被實例化幾次(每個池線程一次)并被緩存和重用。fXt28資訊網——每日最新資訊28at.com

但是虛擬線程永遠不會被池化,也不會被不相關的任務重用。因為每個任務都有自己的虛擬線程,所以每次從不同任務調用 foo 都會觸發新 SimpleDateFormat 的實例化。而且由于可能有大量的虛擬線程同時運行,昂貴的對象可能會消耗相當多的內存。這些結果與線程本地緩存想要實現的結果恰恰相反。fXt28資訊網——每日最新資訊28at.com

對于線程局部變量緩存可重用對象的問題,沒有什么好的通用替代方案,但對于 SimpleDateFormat,我們應該將其替換為 DateTimeFormatter。DateTimeFormatter 是不可變的,因此單個實例就可以由所有線程共享:fXt28資訊網——每日最新資訊28at.com

static final DateTimeFormatter formatter = DateTimeFormatter….;void foo() {  ... formatter.format(...); ...}

需要注意的是,使用線程局部變量來緩存共享的昂貴對象有時是由一些異步框架在幕后完成的,其隱含的假設是這些可重用對象只會由極少數池線程使用。fXt28資訊網——每日最新資訊28at.com

所以混合虛擬線程和異步框架一起使用可能不是一個好主意,對某些方法的調用可能會導致可重用對象被重復創建。fXt28資訊網——每日最新資訊28at.com

避免長時間和頻繁的 synchronized

當前虛擬線程實現由一個限制是,在同步塊或方法內執行 synchronized 阻塞操作會導致 JDK 的虛擬線程調度程序阻塞寶貴的操作系統線程,而如果阻塞操作是在同步塊或方法外完成的,則不會被阻塞。我們稱這種情況為 “Pinning”。fXt28資訊網——每日最新資訊28at.com

如果阻塞操作既長期又頻繁,則 “Pinning” 可能會對服務器的吞吐量產生不利影響。如果阻塞操作短暫(例如內存中操作)或不頻繁則可能不會產生不利影響。fXt28資訊網——每日最新資訊28at.com

為了檢測可能有害的 “Pinning” 實例,(JDK Flight Recorder (JFR) 在 “Pinning” 阻塞時間超過 20 毫秒時,會發出 jdk.VirtualThreadPinned 事件。fXt28資訊網——每日最新資訊28at.com

或者我們可以使用系統屬性 jdk.tracePinnedThreads 在線程被 “Pinning” 阻塞時發出堆棧跟蹤。fXt28資訊網——每日最新資訊28at.com

啟動 Java 程序時添加 -Djdk.tracePinnedThreads=full 運行,會在線程被 “Pinning” 阻塞時打印完整的堆棧跟蹤,突出顯示本機幀和持有監視器的幀。使用 -Djdk.tracePinnedThreads=short 運行,會將輸出限制為僅有問題的幀。fXt28資訊網——每日最新資訊28at.com

如果這些機制檢測到既長期又頻繁 “Pinning” 的地方,請在這些特定地方將 synchronized 替換為 ReentrantLock。以下是長期且頻繁使用 synchronized 的示例:fXt28資訊網——每日最新資訊28at.com

synchronized(lockObj) {    frequentIO();}

我們可以將其替換為以下內容:fXt28資訊網——每日最新資訊28at.com

lock.lock();try {    frequentIO();} finally {    lock.unlock();}

參考資料:https://docs.oracle.com/en/java/javase/21/core/virtual-threads.html#GUID-E695A4C5-D335-4FA4-B886-FEB88C73F23EfXt28資訊網——每日最新資訊28at.com

最后說兩句

針對虛擬線程的使用,相信大家心里已經有了答案。在對虛擬線程需要限制并發數的場景,使用信號量即可。在虛擬線程中使用線程局部變量時要注意避免緩存昂貴的可重用對象。對于使用到 synchronized 同步塊或者方法的虛擬線程,建議替換為 ReentrantLock,避免影響吞吐量。fXt28資訊網——每日最新資訊28at.com

本文鏈接:http://m.www897cc.com/showinfo-26-60959-0.htmlJava 新技術:虛擬線程使用指南

聲明:本網頁內容旨在傳播知識,若有侵權等問題請及時與本網聯系,我們將在第一時間刪除處理。郵件:2376512515@qq.com

上一篇: 接手了個項目,被if..else搞懵逼了

下一篇: 十個優秀免費開源CRM項目

標簽:
  • 熱門焦點
Top 日韩成人免费在线_国产成人一二_精品国产免费人成电影在线观..._日本一区二区三区久久久久久久久不
亚洲电影在线| 国产日韩视频| 欧美精品国产精品| 欧美日韩一区在线观看视频| 国产精品欧美日韩久久| 国产最新精品精品你懂的| 亚洲人成在线播放| 亚洲一区二区不卡免费| 久久福利资源站| 欧美v国产在线一区二区三区| 欧美日韩一区在线视频| 国产亚洲精品激情久久| 男女精品网站| 国产精品www994| 伊人精品视频| 亚洲一级黄色片| 美女性感视频久久久| 欧美日韩一区三区| 韩日精品视频| 在线综合欧美| 美女在线一区二区| 国产精品久久看| 亚洲国产精品一区二区www在线| 亚洲午夜黄色| 欧美91视频| 国产日韩欧美不卡| 一本色道久久综合亚洲精品婷婷| 欧美伊人久久久久久午夜久久久久 | 亚洲欧美日韩精品久久奇米色影视| 久久婷婷麻豆| 国产精品欧美久久久久无广告| 亚洲风情在线资源站| 亚洲欧美日韩精品久久亚洲区 | 欧美在线视频免费| 欧美日韩国产精品一区二区亚洲| 狠狠综合久久av一区二区老牛| 在线亚洲伦理| 欧美激情影院| 一区在线视频| 久久aⅴ国产欧美74aaa| 欧美性生交xxxxx久久久| 影音先锋另类| 欧美一区二区三区婷婷月色| 欧美日韩国产二区| 蜜臀av性久久久久蜜臀aⅴ四虎| 激情久久中文字幕| 男女av一区三区二区色多| 雨宫琴音一区二区在线| 久久免费一区| 国产一区二区按摩在线观看| 午夜一区二区三区在线观看| 欧美视频精品一区| 日韩一级大片在线| 欧美视频不卡中文| 亚洲欧美清纯在线制服| 伊人天天综合| 亚洲女性裸体视频| 国产乱码精品| 欧美日韩专区在线| 午夜精品视频网站| 国产毛片一区| 午夜一区二区三视频在线观看| 国产精品白丝av嫩草影院| 亚洲影院一区| 中文国产成人精品| 亚洲日本精品国产第一区| 国产精品人人爽人人做我的可爱| 亚洲欧美中文在线视频| 最近看过的日韩成人| 国产精品亚洲精品| 欧美制服丝袜第一页| 黄色精品在线看| 久久综合久久久久88| 午夜精品婷婷| 国产日韩一区在线| 国产午夜亚洲精品不卡| 国产精品激情偷乱一区二区∴| 亚洲欧美在线高清| 亚洲色图自拍| 一区二区精品| 欧美一级夜夜爽| 裸体一区二区| 免费在线看成人av| 欧美视频网址| 国产精品久久久久一区二区| 欧美日本国产| 国产精品日韩精品欧美精品| 国产一区二区三区奇米久涩| 亚洲视屏在线播放| 乱中年女人伦av一区二区| 久久亚洲国产精品日日av夜夜| 一本色道久久精品| 日韩视频永久免费观看| 亚洲先锋成人| 久久天堂成人| 亚洲福利视频网| 欧美大片在线观看| 亚洲免费福利视频| 欧美午夜精品久久久久久浪潮| 夜夜嗨av一区二区三区网站四季av| 欧美日韩在线不卡一区| 亚洲午夜免费视频| 国产精品一页| 久久久久九九九| 亚洲国产精品成人一区二区| 欧美国产1区2区| 一二三区精品| 国产麻豆精品久久一二三| 久久精品综合网| 亚洲激情偷拍| 亚洲宅男天堂在线观看无病毒| 国产精品久久久久久久久借妻| 欧美影院视频| 亚洲国产精品久久91精品| 欧美日韩国产成人| 亚洲欧洲av一区二区| 一区二区三区在线免费播放| 欧美黄色成人网| 亚洲一区二区动漫| 伊人久久大香线蕉av超碰演员| 欧美激情一区二区三区| 亚洲永久免费| 在线观看视频亚洲| 国产精品hd| 久久精品国产77777蜜臀| 亚洲激情视频网站| 国产精品久久久免费| 久久综合伊人77777尤物| 一区二区久久久久| 国产亚洲欧美色| 欧美激情一区二区三区全黄 | 欧美人与禽猛交乱配视频| 亚洲——在线| 尤物网精品视频| 国产精品超碰97尤物18| 久久青草欧美一区二区三区| 一本色道久久88综合日韩精品| 国产欧美在线观看一区| 麻豆精品精品国产自在97香蕉| 一区二区激情| 黑人巨大精品欧美一区二区| 欧美日韩色婷婷| 久久久国产午夜精品| 在线亚洲国产精品网站| 精品成人在线观看| 国产精品mm| 欧美3dxxxxhd| 欧美在线一二三区| 野花国产精品入口| 好吊色欧美一区二区三区视频| 欧美日韩在线免费视频| 久久综合色婷婷| 午夜精品一区二区三区电影天堂| 91久久精品国产91性色tv| 国产欧美精品在线播放| 欧美精品在线看| 久久精品国产免费看久久精品| 一区二区三区鲁丝不卡| 亚洲国产欧美一区二区三区久久 | 亚洲视频一区在线观看| 在线视频成人| 国产亚洲激情视频在线| 国产精品a久久久久| 欧美成人资源| 久久久久久久尹人综合网亚洲| 亚洲一区中文| 日韩系列在线| 亚洲高清免费| 久久人人爽人人爽| 亚洲在线观看| 一本色道久久加勒比精品| 91久久精品国产91久久性色| 国内精品99| 国产区二精品视| 国产精品捆绑调教| 欧美色视频在线| 欧美精品一区二区三区四区| 麻豆国产精品777777在线| 久久精品成人一区二区三区| 亚洲女人天堂成人av在线| 一区二区三区三区在线| 国产欧美精品一区二区色综合 | 亚洲黄一区二区| 狠狠入ady亚洲精品| 国产精品自拍网站| 国产精品电影网站| 欧美日韩在线精品| 欧美人成免费网站| 欧美精品色网| 欧美精品乱码久久久久久按摩| 美女视频一区免费观看| 久久亚洲精品一区二区| 久久久777| 久久er99精品| 久久av老司机精品网站导航| 午夜久久久久久| 亚洲欧美日韩国产一区| 亚洲一区二区三区四区五区黄| 一区二区三区欧美亚洲| 国产亚洲精品自拍| 国产一区二区精品丝袜| 国产亚洲成年网址在线观看| 国产日韩欧美不卡在线| 国产三级精品在线不卡|