国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

提升系統(tǒng)吞吐量,詳解JDK21虛擬線程,炸裂

開發(fā) 前端
當前實現虛擬線程的一個限制是,在同步的塊或方法內部執(zhí)行阻塞操作會導致JDK的虛擬線程調度器阻塞一個操作系統(tǒng)線程,而在同步的塊或方法外部執(zhí)行阻塞操作則不會。

環(huán)境:JDK21

1. 虛擬線程簡介

虛擬線程是輕量級的線程,可以減少編寫、維護和調試高吞吐量并發(fā)應用程序的工作量。線程是可以調度的最小處理單元。它與其他類似單元并發(fā)運行,而且在很大程度上是獨立運行的。它是java.lang.Thread的一個實例。線程有兩種,平臺線程和虛擬線程。

2. 什么是平臺線程

平臺線程被實現為操作系統(tǒng)(OS)線程的薄包裝器。平臺線程在其底層操作系統(tǒng)線程上運行Java代碼,平臺線程在平臺線程的整個生命周期中捕獲其操作系統(tǒng)線程。因此,可用平臺線程的數量受限于操作系統(tǒng)線程的數量。

平臺線程通常有一個比較大的線程堆棧和由操作系統(tǒng)維護的其他資源。它們適用于運行所有類型的任務,但可能是有限的資源。

3. 什么是虛擬線程

與平臺線程一樣,虛擬線程也是java.lang.thread的一個實例。然而,虛擬線程并沒有綁定到特定的操作系統(tǒng)線程。虛擬線程仍然在操作系統(tǒng)線程上運行代碼。但是,當虛擬線程中運行的代碼調用阻塞I/O操作時,Java運行時會掛起虛擬線程,直到可以恢復為止。與掛起的虛擬線程相關聯(lián)的OS線程現在可以自由地執(zhí)行其他虛擬線程的操作。

虛擬線程的實現方式與虛擬內存類似。為了模擬大量內存,操作系統(tǒng)將大量虛擬地址空間映射到有限的RAM。同樣,為了模擬大量線程,Java運行時將大量虛擬線程映射到少量操作系統(tǒng)線程。

與平臺線程不同,虛擬線程通常有一個淺調用堆棧,只執(zhí)行一個HTTP客戶端調用或一個JDBC查詢。盡管虛擬線程支持線程本地變量和可繼承的線程本地變量,但應該仔細考慮使用它們,因為單個JVM可能支持數百萬個虛擬線程。

虛擬線程適用于運行大部分時間被阻塞的任務,這些任務通常等待I/O操作完成。然而,它們并不適用于長時間運行的CPU密集型操作。

4. 為什么使用虛擬線程

在高吞吐量并發(fā)應用程序中使用虛擬線程,尤其是那些由大量并發(fā)任務組成、花費大量時間等待的應用程序。服務器應用程序是高吞吐量應用程序的示例,因為它們通常處理許多執(zhí)行阻塞I/O操作(如獲取資源)的客戶端請求。

虛擬線程不是更快的線程;它們運行代碼的速度并不比平臺線程快。它們的存在是為了提供規(guī)模(更高的吞吐量),而不是速度(更低的延遲)。

5. 創(chuàng)建虛擬線程

Thread和Thread.Builder APIs提供了創(chuàng)建平臺線程和虛擬線程的方法。java.util.concurrent.Executors類還定義了創(chuàng)建ExecutorService的方法,該方法為每個任務啟動一個新的虛擬線程。

5.1 Thread & Thread.Builder創(chuàng)建虛擬線程

調用Thread.ofVirtual()方法創(chuàng)建一個Thread.Builder實例,用于創(chuàng)建虛擬線程。如下示例:

Thread t= Thread.ofVirtual().start(() -> System.out.println("Hello")) ;
t.join() ;

Thread.Builder接口允許創(chuàng)建具有公共線程屬性(如線程名稱)的線程。Thread.Builder.OfPlatform子接口創(chuàng)建平臺線程,而Thread.Builder.OfVirtual創(chuàng)建虛擬線程。

下面的示例使用Thread.Builder接口創(chuàng)建一個名為T-VM的虛擬線程,如下示例:

Thread.Builder builder = Thread.ofVirtual().name("T-VM") ;
Runnable task = () -> {
  System.out.println("執(zhí)行任務") ;
} ;
Thread t = builder.start(task) ;
System.err.printf("線程名稱: %s%n", t.getName()) ;
t.join() ;

輸出結果:

執(zhí)行任務
線程名稱T-VM

下面的示例創(chuàng)建并啟動兩個具有Thread.Builder的虛擬線程:

Thread.Builder builder = Thread.ofVirtual().name("vm - worker - ", 0);
Runnable task = () -> {
  System.out.printf("線程ID: %d%n", Thread.currentThread().threadId());
} ;
// 線程 "vm - worker - 0"
Thread t1 = builder.start(task) ;   
t1.join();
System.out.println(t1.getName() + " terminated") ;
// 線程 "vm - worker - 1"
Thread t2 = builder.start(task) ;   
t2.join() ;  
System.out.println(t2.getName() + " terminated") ;

輸出結果:

線程ID: 21
vm - worker - 0 terminated
線程ID: 24
vm - worker - 1 terminated

以上是通過Thread.Builder創(chuàng)建虛擬線程的簡單示例。

5.2 Executors創(chuàng)建虛擬線程

Executors允許將線程管理和創(chuàng)建與應用程序的其余部分分離。

ExecutorService executor = Executors.newVirtualThreadPerTaskExecutor() ;
// submit Runnable任務
Future<?> future = executor.submit(() -> System.out.println("Running thread")) ;
future.get() ;
System.out.println("Task completed") ;

上面示例每當調用ExecutorService.submit(Runnable)時,都會創(chuàng)建一個新的虛擬線程并開始運行該任務。

6. 虛擬線程調度

操作系統(tǒng)在平臺線程運行時進行調度。然而,Java運行時會在虛擬線程運行時進行調度。當Java運行時調度虛擬線程時,它將虛擬線程分配或掛載到平臺線程上,然后操作系統(tǒng)像往常一樣調度該平臺線程。這個平臺線程稱為載體(carrier)。運行一些代碼后,虛擬線程可以從它的載體卸載。這通常發(fā)生在虛擬線程執(zhí)行阻塞I/O操作時。當一個虛擬線程從其宿主中卸載后,宿主就處于空閑狀態(tài),這意味著Java運行時調度器可以在其上分配另一個虛擬線程。

虛擬線程被綁定到其宿主(平臺線程)時,在阻塞操作期間無法將其卸載。虛擬線程在以下情況下會被綁定:

  • 虛擬線程在同步塊或方法內運行代碼
  • 虛擬線程運行本機方法或外部函數

7. 虛擬線程應用指南

虛擬線程是由Java運行時而不是操作系統(tǒng)實現的Java線程。虛擬線程和傳統(tǒng)線程(平臺線程)的主要區(qū)別在于,可以很容易地在同一個Java進程中運行大量甚至數百萬個活動的虛擬線程。正是它們的數量賦予了虛擬線程強大的能力,通過允許服務器并發(fā)處理更多請求,它們可以更高效地運行以"thread-per-request"風格編寫的服務器應用程序,從而提高吞吐量,減少硬件浪費。

虛擬線程可以顯著提高以thread-per-request風格編寫的服務器的吞吐量,而不是延遲。在這種風格中,服務器在整個持續(xù)時間內使用一個線程來處理每個傳入的請求。它至少占用一個線程,因為在處理單個請求時,你可能希望使用更多線程并發(fā)地執(zhí)行某些任務。

阻塞平臺線程的代價是昂貴的,因為它會占用線程——這是一種相對稀缺的資源——而線程并沒有做很多有意義的工作。因為虛擬線程可能很多,所以阻塞它們是廉價的,也是值得鼓勵的。因此,應該使用簡單的同步風格并使用阻塞I/O API編寫代碼。

如下代碼以非阻塞異步風格編寫,不會從虛擬線程中獲得太多好處。

HttpClient client = ... ;
Executor pool = Executors.newVirtualThreadPerTaskExecutor() ;
CompletableFuture.supplyAsync(() -> {
  HttpRequest request = HttpRequest.newBuilder(URI.create("http://localhost:8088/users/info")).build() ;
  BodyHandler<String> bodyHandler = ... ;
  try {
    return client.send(request , bodyHandler) ;
  }
}, pool)
.thenCompose(url -> getBodyAsync(url, HttpResponse.BodyHandlers.ofString()))
.thenApply(info::findImage)
.thenAccept(this::process)
.exceptionally(t -> { t.printStackTrace(); return null; });

相反,以下代碼以同步風格編寫,并使用簡單的阻塞IO,將受益匪淺:

try {
   String page = getBody(info.getUrl(), HttpResponse.BodyHandlers.ofString());
   String imageUrl = info.findImage(page);
   byte[] data = getBody(imageUrl, HttpResponse.BodyHandlers.ofByteArray());   
   info.setImageData(data);
   process(info);
}

8. 不要池化虛擬線程

關于虛擬線程最難理解的是,雖然它們與平臺線程具有相同的行為,但它們不應該表示相同的程序概念。

平臺線程很少,因此是一種寶貴的資源。寶貴的資源需要管理,管理平臺線程的最常見方法是使用線程池。接下來你需要回答的問題是,線程池中應該有多少個線程?

但是虛擬線程是很多的,因此每個線程不應該代表某種共享的、池化的資源,而應該代表一個任務。

將n個平臺線程轉換為n個虛擬線程幾乎沒有好處;相反,需要轉換的是任務。

為了將每個應用任務表示為一個線程,不要像下面的例子那樣使用共享線程池:

Future<ResultA> f1 = sharedThreadPoolExecutor.submit(task1);
Future<ResultB> f2 = sharedThreadPoolExecutor.submit(task2);
// ... use futures

相反,使用虛擬線程

try (var executor = Executors.newVirtualThreadPerTaskExecutor()) {
   Future<ResultA> f1 = executor.submit(task1);
   Future<ResultB> f2 = executor.submit(task2);
   // ... use futures
}

上面的代碼仍然使用ExecutorService,但從Executors.newVirtualThreadPerTaskExecutor()返回的代碼沒有使用線程池。相反,它為每個提交的任務創(chuàng)建一個新的虛擬線程。

此外,ExecutorService本身是輕量級的,我們可以像創(chuàng)建任何簡單對象一樣創(chuàng)建一個新的對象。這使得我們可以依賴新添加的ExecutorService#close方法和try-with-resources構造。close方法會在try塊結束時隱式調用,它會自動等待所有提交給ExecutorService的任務(即由ExecutorService生成的所有虛擬線程)結束。

對于扇出場景來說,這是一個特別有用的模式,在這種場景中,同時執(zhí)行多個對不同服務調用,如下面的示例所示:

void handle() throws Exception {
  URL url1 = URI.create("http://www.pack.com").toURL() ;
  URL url2 = URI.create("http://www.akf.com").toURL() ;
  try (var executor = Executors.newVirtualThreadPerTaskExecutor()) {
    var future1 = executor.submit(() -> fetchURL(url1));
    var future2 = executor.submit(() -> fetchURL(url2));
    System.out.printf("result1: %s, result2: %s%n", future1.get(),future2.get()) ;
  }
}


String fetchURL(URL url) throws IOException {
  try (var in = url.openStream()) {
    return new String(in.readAllBytes(), StandardCharsets.UTF_8);
  }
}

你應該創(chuàng)建一個新的虛擬線程,如上所示,即使是小的、短期的并發(fā)任務。

9. 避免長時間頻繁的Pinning

當前實現虛擬線程的一個限制是,在同步的塊或方法內部執(zhí)行阻塞操作會導致JDK的虛擬線程調度器阻塞一個操作系統(tǒng)線程,而在同步的塊或方法外部執(zhí)行阻塞操作則不會。這種情況稱為“Pinning”。如果阻塞操作持續(xù)時間長且頻繁,Pinning可能會對服務器的吞吐量產生負面影響。保護短期的操作,例如內存操作,或者使用同步塊或方法的不頻繁操作,應該不會有任何負面影響。

對于長時間又頻繁的地方應該使用ReentrantLock替換synchronized 。

synchronized(lockObj) {
  frequentIO() ;
}
// 替換為
lock.lock();
try {
  frequentIO() ;
} finally {
  lock.unlock() ;
}


責任編輯:武曉燕 來源: Spring全家桶實戰(zhàn)案例源碼
相關推薦

2023-11-03 18:23:34

虛擬線程服務器

2023-10-20 08:12:00

JDK21線程池配置

2024-05-23 16:41:40

2023-12-28 10:49:27

響應式編程異步

2023-02-09 08:57:11

Callable異步java

2025-09-11 02:00:00

2023-09-20 09:07:01

Java 21開發(fā)工具包

2010-04-14 16:02:09

IDF

2024-09-12 15:24:29

2013-04-19 09:45:20

AMPLabHadoopHDFS

2024-09-09 14:12:38

2021-12-26 00:03:27

響應式編程異步

2025-05-09 02:00:00

代碼接口吞吐量

2024-01-19 13:42:00

模型訓練

2025-06-13 09:12:28

2025-03-04 08:52:21

2024-12-13 13:58:53

2024-11-14 15:00:00

線程架構吞吐量

2023-08-03 14:18:29

Rust阻塞函數

2023-11-07 15:11:46

Kafka技巧
點贊
收藏

51CTO技術棧公眾號

久久夜色精品国产噜噜av| 精品肉辣文txt下载| 成人美女视频在线观看18| 成人亚洲性情网站www在线观看| 三级久久三级久久久| 樱空桃在线播放| 国模少妇一区二区三区| 精品视频无码一区二区三区| 午夜精品123| 高清电影在线免费观看| 欧美国产精品日韩| 亚洲老妇激情| 波多野结衣激情| 亚洲精品国产高清久久伦理二区| 国产区视频在线播放| 国产亚洲精品91在线| 欧美疯狂party性派对| 青青草综合视频| 亚洲成av人影院| 亚州一区二区三区| 91久久国产婷婷一区二区| 国产精品一区二区果冻传媒| 免费黄色网页| 亚洲人成啪啪网站| 无码一区二区三区视频| 国产毛片视频网站| 色综合中文字幕国产| 欧美久久久网站| 国产一级精品aaaaa看| 久久亚洲二区三区| 日本中文字幕在线观看| 欧美成人午夜激情在线| 亚洲激情综合| 国产免费视频| 亚洲人成在线播放| 午夜欧美理论片| 国产97色在线 | 日韩| 日韩你懂的在线观看| 精品国精品国产自在久国产应用| 精品人妻大屁股白浆无码| 色婷婷国产精品久久包臀| 清纯唯美激情亚洲| 在线视频欧美一区| 欧美在线一区二区三区| 亚洲制服一区| 免费av观看网址| 欧美xxxx老人做受| 久久一区二区三区电影| 国产午夜福利视频在线观看| 在线成人免费视频| 日韩欧美综合| 天天爱天天做色综合| 一本色道久久综合狠狠躁篇的优点| 亚洲茄子视频| 桥本有菜亚洲精品av在线| 九九久久精品一区| 国产伦精一区二区三区| 色呦呦久久久| 精品视频第一区| 日韩欧美国产网站| 成人羞羞视频播放网站| 成人福利网址| 久久成年人免费电影| 国产精品538一区二区在线| 国产cdts系列另类在线观看| 亚洲自拍偷拍福利| 午夜久久久久久| 国产一区二区电影在线观看| 亚洲这里只有精品| 欧美美最猛性xxxxxx| 成人丝袜视频网| av资源中文在线天堂| 蜜桃久久精品乱码一区二区 | 精品电影一区| 一级毛片免费视频| 91tv亚洲精品香蕉国产一区7ujn| 久久久午夜精品理论片中文字幕| 欧美大片网站| 日韩av三级在线| 久久夜色精品国产| 久久精品一区蜜桃臀影院| 亚洲青青一区| 久久久久狠狠高潮亚洲精品| 美女视频久久黄| 国产亚洲欧美日韩俺去了| 四虎地址8848精品| 欧美污视频网站| 久久久久久18| 亚洲男人的天堂网| 欧美激情电影| h网站视频在线观看| 国产一区二区三区四区五区加勒比| 欧美中文字幕亚洲一区二区va在线| 欧美精品国产一区二区| avtt在线播放| 三级三级久久三级久久18| 亚洲国产婷婷香蕉久久久久久| 国产一区在线不卡| 四虎影视国产精品| 国产91白丝在线播放| 国产成人午夜视频网址| 婷婷开心激情综合| 欧美日韩一区自拍| 操你啦视频在线| 最新精品视频| 久久成人精品视频| 亚洲精品成人精品456| 国产精品videosex性欧美| 二区在线观看| 欧美视频观看一区| 亚洲欧美精品中文字幕在线| 不卡av在线网| 精品中文一区| 91在线不卡| 成年在线观看视频| 992tv成人免费影院| 福利一区视频在线观看| 99日韩精品| 亚洲欧美在线成人| 免费在线黄色av| 欧美日韩亚洲一区二区三区在线观看 | 中文字幕精品—区二区| 国产女人aaa级久久久级| 97视频精品| √天堂8资源中文在线| 日本黄网站免费| 国产女同一区二区| 欧美xxxx老人做受| 国产精品久久久久久妇女6080| 欧美在线三级| 欧美18—19sex性hd| 嫩草懂你的影院| 国产日韩一区二区三区| 日韩专区在线播放| 黑人极品videos精品欧美裸| 黄色小说综合网站| 九九热精品视频在线观看| 日本无删减在线| 黄网免费入口| 五月天亚洲综合| 欧洲亚洲免费视频| 亚洲黄页视频免费观看| 一区二区三区在线视频观看| 日日摸夜夜添夜夜添亚洲女人| av不卡一区| 性欧美videos高清hd4k| 激情综合网五月激情 | 日韩精品免费一区二区三区竹菊| 狠狠狠综合7777久夜色撩人| 超薄肉色丝袜足j调教99| 国产精国产精品| 日韩精品视频中文在线观看 | 国产在线拍揄自揄视频不卡99| 亚洲第一男人av| 亚洲女与黑人做爰| 九一久久久久久| 97精品国产福利一区二区三区| 欧美人体一区二区三区| 九色在线播放| 欧美性大战久久久久xxx| 国产精品区一区| 97在线免费观看视频| 日韩精品中文字| 欧美午夜在线观看| 亚洲日本韩国一区| k8久久久一区二区三区| 免费国产自线拍一欧美视频| 国产区精品区| **精品中文字幕一区二区三区| av毛片在线播放| 在线影院av| youjizzxxxx18| 在线亚洲美日韩| 1卡2卡3卡精品视频| 97免费在线视频| 在线色欧美三级视频| 日韩欧美一级在线播放| 欧美日韩在线免费观看| 亚洲视频免费在线| 91麻豆6部合集magnet| 久久99精品久久久久久久久久久久| 国产精品久久久久蜜臀| 澳门精品久久国产| 免费高清视频在线一区| av中文字幕在线看| 欧美激情午夜| 国产高清美女一级毛片久久| 亚洲美女在线免费观看| 先锋影音资源综合在线播放av| 欧美一级免费播放| 穿情趣内衣被c到高潮视频| 欧美一区二区三区四区在线观看地址| 91久热免费在线视频| 国产成人91久久精品| 久久久久久久久久久av| 久久av.com| 久久久久久久一区二区| 欧美夫妻性视频| 欧美乱妇40p| 久久久久久com|