国产 无码 综合区,色欲AV无码国产永久播放,无码天堂亚洲国产AV,国产日韩欧美女同一区二区

Java 21 虛擬線程:使用指南(一)

這篇具有很好參考價值的文章主要介紹了Java 21 虛擬線程:使用指南(一)。希望對大家有所幫助。如果存在錯誤或未考慮完全的地方,請大家不吝賜教,您也可以點擊"舉報違法"按鈕提交疑問。

虛擬線程是由 Java 21 版本中實現(xiàn)的一種輕量級線程。它由 JVM 進行創(chuàng)建以及管理。虛擬線程和傳統(tǒng)線程(我們稱之為平臺線程)之間的主要區(qū)別在于,我們可以輕松地在一個 Java 程序中運行大量、甚至數(shù)百萬個虛擬線程。

由于虛擬線程的數(shù)量眾多,也就賦予了 Java 程序強大的力量。虛擬線程適合用來處理大量請求,它們可以更有效地運行 “一個請求一個線程” 模型編寫的 web 應用程序,可以提高吞吐量以及減少硬件浪費。

由于虛擬線程是 java.lang.Thread 的實現(xiàn),并且遵守自 Java SE 1.0 以來指定 java.lang.Thread 的相同規(guī)則,因此開發(fā)人員無需學習新概念即可使用它們。

但是虛擬線程才剛出來,對我們來說有一些陌生。由于 Java 歷來版本中無法生成大量平臺線程(多年來 Java 中唯一可用的線程實現(xiàn)),已經(jīng)讓程序員養(yǎng)成了一套關于平臺線程的使用習慣。這些習慣做法在應用于虛擬線程時會適得其反,我們需要摒棄。

此外虛擬線程和平臺線程在創(chuàng)建成本上的巨大差異,也提供了一種新的關于線程使用的方式。Java 的設計者鼓勵使用虛擬線程而不必擔心虛擬線程的創(chuàng)建成本。

本文無意全面涵蓋虛擬線程的每個重要細節(jié),目的只是提供一套介紹性指南,以幫助那些希望開始使用虛擬線程的人充分利用它們。

本文完整大綱如下,

Java 21 虛擬線程:使用指南(一)

請大方使用同步阻塞 IO

虛擬線程可以顯著提高以 “一個請求一個線程” 模型編寫的 web 應用程序的吞吐量(注意不是延遲)。在這種模型中,web 應用程序針對每個客戶端請求都會創(chuàng)建一個線程進行處理。因此為了處理更多的客戶端請求,我們需要創(chuàng)建更多的線程。

在 “一個請求一個線程” 模型中使用平臺線程的成本很高,因為平臺線程與操作系統(tǒng)線程對應(操作系統(tǒng)線程是一種相對稀缺的資源),阻塞了平臺線程,會讓它無事可做一直處于阻塞中,這樣就會造成很大的資源浪費。

然而,在這個模型中使用虛擬線程就很合適,因為虛擬線程非常廉價就算被阻塞也不會造成資源浪費。因此在虛擬線程出來后,Java 的設計者是建議我們應該以簡單的同步風格編寫代碼并使用阻塞 IO。

舉個例子,以下用非阻塞異步風格編寫的代碼是不會從虛擬線程中受益太多的,

CompletableFuture.supplyAsync(info::getUrl, pool)
   .thenCompose(url -> getBodyAsync(url, HttpResponse.BodyHandlers.ofString()))
   .thenApply(info::findImage)
   .thenCompose(url -> getBodyAsync(url, HttpResponse.BodyHandlers.ofByteArray()))
   .thenApply(info::setImageData)
   .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);
} catch (Exception ex) {
   t.printStackTrace();
}

并且上面的同步代碼也更容易在調試器中調試、在分析器中分析或通過線程轉儲進行觀察。要觀察虛擬線程,可以使用 jcmd 命令創(chuàng)建線程轉儲,

jcmd <pid> Thread.dump_to_file -format=json <file>

用同步風格并使用阻塞 IO 風格編寫的代碼越多,虛擬線程的性能和可觀察性就越好。而用異步非阻塞 IO 風格編寫的程序或框架,如果每個任務沒有專用一個線程,則無法從虛擬線程中獲得顯著的好處。

使用虛擬線程,我們因該避免將同步阻塞 IO 與異步非阻塞 IO 混為一談。

避免池化虛擬線程

關于虛擬線程使用方面最難理解的一件事情就是,我們不應該池化虛擬線程。雖然虛擬線程具有與平臺線程相同的行為,但虛擬線程和線程池其實是兩種概念。

平臺線程是一種稀缺資源,因為它很寶貴。越寶貴的資源就越需要管理,管理平臺線程最常見的方法是使用線程池。

不過在使用線程池后,我們需要回答的一個問題,線程池中應該有多少個線程?最小線程數(shù)、最大線程數(shù)應該設置多少?這也是一個問題。

虛擬線程是一種非常廉價的資源,每個虛擬線程不應代表某些共享的、池化的資源,而應代表單一任務。在應用程序中,我們應該直接使用虛擬線程而不是通過線程池使用它。

那么我們應該創(chuàng)建多少個虛擬線程嘞?答案是不必在乎虛擬線程的數(shù)量,我們有多少個并發(fā)任務就可以有多少個虛擬線程。

如下是一段提交任務的代碼,將每個任務都提交到線程池中執(zhí)行,在 Java 21 以后,不建議再使用共享線程池執(zhí)行器,代碼如下,

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

建議使用虛擬線程執(zhí)行器,代碼如下,

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

上面代碼雖然仍使用 ExecutorService,但從 Executors.newVirtualThreadPerTaskExecutor() 方法返回的執(zhí)行器不再使用線程池。它會為每個提交的任務都創(chuàng)建一個新的虛擬線程。

此外,ExecutorService 本身是輕量級的,我們可以像創(chuàng)建任何簡單對象一樣直接創(chuàng)建一個新的 ExecutorService 對象而不必考慮復用。

這使我們能夠依賴 Java 19 中新添加的 ExecutorService.close() 方法和 try-with-resources 語法糖。在 try 塊末尾隱式調用 ExecutorService.close() 方法,會自動等待提交給 ExecutorService 的所有任務(即 ExecutorService 生成的所有虛擬線程)終止。

對于廣播場景來說,使用 Executors.newVirtualThreadPerTaskExecutor() 比較合適,在這種場景中,希望同時對不同的服務執(zhí)行多個傳出調用,并且方法結束時就關閉線程池,代碼如下,

void handle(Request request, Response response) {
    var url1 = ...
    var url2 = ...

    try (var executor = Executors.newVirtualThreadPerTaskExecutor()) {
        var future1 = executor.submit(() -> fetchURL(url1));
        var future2 = executor.submit(() -> fetchURL(url2));
        response.send(future1.get() + future2.get());
    } catch (ExecutionException | InterruptedException e) {
        response.fail(e);
    }
}

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

針對廣播模式和其他常見的并發(fā)模式,如果希望有更好的可觀察性,建議使用結構化并發(fā)。這是 Java 21 中新出的特性,這里給大家賣個關子,我將在后續(xù)進行講解。

根據(jù)經(jīng)驗來說,如果我們的應用程序從未經(jīng)歷 1 萬的并發(fā)訪問,那么它不太可能從虛擬線程中受益。一方面它負載太輕而不需要更高的吞吐量,一方面并發(fā)請求任務也不夠多。

參考資料

  • https://docs.oracle.com/en/java/javase/21/core/virtual-threads.html#GUID-E695A4C5-D335-4FA4-B886-FEB88C73F23E

最后說兩句

針對虛擬線程的使用,相信大家心里已經(jīng)有了答案。虛擬線程不同于平臺線程,它非常廉價,Java 的設計者鼓勵我們直接使用虛擬線程,而無需池化,也不必擔心過多的虛擬現(xiàn)場會影響性能。

事實上,虛擬現(xiàn)場就是為了解決同步阻塞 IO 對硬件的資源利用率不夠高這一問題。

關注公眾號【waynblog】每周分享技術干貨、開源項目、實戰(zhàn)經(jīng)驗、國外優(yōu)質文章翻譯等,您的關注將是我的更新動力!文章來源地址http://www.zghlxwxcb.cn/news/detail-763090.html

到了這里,關于Java 21 虛擬線程:使用指南(一)的文章就介紹完了。如果您還想了解更多內(nèi)容,請在右上角搜索TOY模板網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關文章,希望大家以后多多支持TOY模板網(wǎng)!

本文來自互聯(lián)網(wǎng)用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。如若轉載,請注明出處: 如若內(nèi)容造成侵權/違法違規(guī)/事實不符,請點擊違法舉報進行投訴反饋,一經(jīng)查實,立即刪除!

領支付寶紅包贊助服務器費用

相關文章

  • Java 21 新特性:虛擬線程(Virtual Threads)

    在Java 21中,引入了虛擬線程(Virtual Threads)來簡化和增強并發(fā)性,這使得在Java中編程并發(fā)程序更容易、更高效。 虛擬線程,也稱為“用戶模式線程(user-mode threads)”或“纖程(fibers)”。該功能旨在簡化并發(fā)編程并提供更好的可擴展性。虛擬線程是輕量級的,這意味著它

    2024年02月08日
    瀏覽(19)
  • Java 21 正式 GA,虛擬線程真的來了

    UTC 時間 2023 年 9 月 19 日,期盼已久的 Java 21 終于發(fā)布正式版! 本文一起來看看其中最受 Java 開發(fā)者關注的一項新特性:Loom 項目的兩個新特性之一的 ”虛擬線程(Virtual Thread)“(另外一個新特性是 ”結構化并發(fā)(Structured Concurrency)“,當前是預覽狀態(tài)),它被稱之為 J

    2024年02月08日
    瀏覽(22)
  • Java21上手體驗-分代ZGC和虛擬線程

    幾天前Oracle剛剛發(fā)布了Java21, 由于這是最新的LTS版本,引起了大家的關注。 我也第一時間在個人項目中進行了升級體驗。 一探究竟,和大家分享。 https://jdk.java.net/21/release-notes https://my.oschina.net/waylau/blog/10112170 JEP 431:序列集合 JEP 439:分代 ZGC JEP 440:記錄模式 JEP 441:switc

    2024年02月08日
    瀏覽(22)
  • Java 21 虛擬線程如何限流控制吞吐量

    虛擬線程(Virtual Threads)是 Java 21 所有新特性中最為吸引人的內(nèi)容,它可以大大來簡化和增強Java應用的并發(fā)性。但是,隨著這些變化而來的是如何最好地管理此吞吐量的問題。本文,就讓我們看一下開發(fā)人員在使用虛擬線程時,應該如何管理吞吐量。 在大多數(shù)情況下,開發(fā)

    2024年02月20日
    瀏覽(15)
  • 【Java】JDK 21中的虛擬線程以及其他新特性

    【Java】JDK 21中的虛擬線程以及其他新特性

    ? 目錄 一、字符串模板(String Templates) 二、序列化集合(Sequenced Collections) 三、分代ZGC(Generational ZGC) 四、記錄模式(Record Patterns) 五、Fibers(纖程) 結論 JDK 21是Java開發(fā)工具包的最新版本,它引入了許多令人振奮的新特性,旨在提高開發(fā)人員的生產(chǎn)力和代碼質量。在本

    2024年02月08日
    瀏覽(25)
  • 使用VirtualBox安裝Linux虛擬機,避坑指南

    使用VirtualBox安裝Linux虛擬機,避坑指南

    ??記錄一下自己學習Linux系統(tǒng)的過程。因為本人一直使用的是Windows系統(tǒng),所以需要裝一個虛擬機來創(chuàng)建一個基于Linux內(nèi)核的操作系統(tǒng)。主要分為以下兩個不步驟:第一步,先安裝虛擬機軟件,我用的是VirtualBox,使用其他的也可以;第二步就是用虛擬機軟件安裝Linux系統(tǒng)。 ?

    2023年04月24日
    瀏覽(87)
  • jdk21 虛擬線程原理及使用分享

    jdk21 虛擬線程原理及使用分享

    jdk21已于北京時間9月19日21點正式發(fā)布, 其中引人注目的就是虛擬線程(Virtual Thread)隨之正式發(fā)布, 不再是此前jdk19、jdk20中的預覽版本。 平臺線程 :java傳統(tǒng)的線程是對系統(tǒng)線程的包裝,為了區(qū)別于虛擬線程,因此將通過傳統(tǒng)方式實現(xiàn)的線程叫做平臺線程(Platform Thread) 虛擬線程

    2024年02月04日
    瀏覽(27)
  • 《cuda c編程權威指南》04 - 使用塊和線程索引映射矩陣索引

    《cuda c編程權威指南》04 - 使用塊和線程索引映射矩陣索引

    目錄 1. 解決的問題 2. 分析 3. 方法 4. 代碼示例 利用塊和線程索引,從全局內(nèi)存中訪問指定的數(shù)據(jù)。 通常情況下, 矩陣 是用行優(yōu)先的方法在 全局內(nèi)存 中線性存儲的。如下。 8列6行矩陣(nx,ny)=(8,6)。 這里建立二維網(wǎng)格(2,3)+二維塊(4,2)為例,使用其塊和線程索引映射矩陣

    2024年02月14日
    瀏覽(19)
  • Node.js 版本管理工具 n 使用指南

    Node.js 版本管理工具 n 使用指南

    Node.js 版本更新很快,目前 node v20.x 已經(jīng)發(fā)布,我們在使用時避免不了會需要切換不同的 Node.js 的版本來使用不同版本的特性。 所以就出現(xiàn)了像 windows 上的 nvm ,MacOS 上的 n 工具,本文就介紹一下如何使用 n 管理 Node.js 的版本。 使用 Brew 安裝時,未安裝可以參考 Brew 官網(wǎng)安裝

    2024年02月16日
    瀏覽(18)
  • Open3D Python版本快速安裝和使用指南

    Open3D Python版本快速安裝和使用指南 Open3D是一個開源的3D計算機視覺庫,提供了一系列用于處理和分析3D數(shù)據(jù)的功能,包括點云、幾何體、圖像等。本文將介紹如何在Python環(huán)境中快速安裝和使用Open3D庫,并提供相應的源代碼示例。 第一步:安裝Python和pip 首先,確保你的系統(tǒng)已

    2024年02月07日
    瀏覽(27)

覺得文章有用就打賞一下文章作者

支付寶掃一掃打賞

博客贊助

微信掃一掃打賞

請作者喝杯咖啡吧~博客贊助

支付寶掃一掃領取紅包,優(yōu)惠每天領

二維碼1

領取紅包

二維碼2

領紅包