支持JDK19虛擬線程的web框架,之三:觀察運行中的虛擬線程

来源:https://www.cnblogs.com/bolingcavalry/archive/2023/09/15/17689837.html
-Advertisement-
Play Games

本篇藉助JProfiler工具,從線程的觀察結果去印證官方資料,做到理論結合實踐,讓您領先一步,掌握和瞭解神秘的虛擬線程內幕 ...


歡迎訪問我的GitHub

這裡分類和彙總了欣宸的全部原創(含配套源碼):https://github.com/zq2599/blog_demos

本篇概覽

  • 本篇是《支持JDK19虛擬線程的web框架》系列的第三篇,在前面兩篇咱們一起瞭解和體驗了支持虛擬線程的web服務,功能性能都試過,整個開發過程也完整執行,算是對quarkus和虛擬線程有了初步的瞭解,但也留下兩個問題
  1. 虛擬線程和常規子線程的區別,究竟能不能看出來?前文已經驗證了性能上區別不大,那還有別的方式來觀察和區分嗎?
  2. 能不能稍微深入一點,僅憑一個@RunOnVirtualThread註解就強行寫兩篇博客,實在是太忽悠人了
  • 本文聚焦第一個問題,與大家一起深入瞭解虛擬線程,重點在理論結合實際,將官方資料在實戰中得到印證
  • 至於第二個問題,留待下一篇...

設置

  • 開始深入學習前有個設置需要確認,否則會導致訪問服務失敗,請打開前文開發的quarkus應用,下圖紅色箭頭指向的配置必須存在,且值必須是0.0.0.0
image-20221022153432270
  • 如果沒有上述配置,IDEA啟動的應用就只會監聽127.0.0.1這塊網卡,如此依賴外部測試工具訪問此應用的服務就無法成功
  • 那麼就開始吧:如何直觀地、清楚地看出虛擬線程和常規子線程的區別?

準備工作

  • 工欲善其事.....咱們先把必要的工具裝上:IDEA的JProfiler插件,安裝步驟如下圖

image-20221015164110766

  • 接下來請在自己電腦上安裝JProfiler,註意,這一步必須要做,詳細的安裝和註(po)冊(jie)過程就不寫在本文中了,請自行搜索相關資料
  • 完成上述準備後,點擊下圖箭頭所指按鈕,這樣就指定了JProfiler去監控分析啟動後的應用進程
  • IDEA會拉起JProfiler
image-20221022114521069
  • 出現新的視窗如下圖,再點擊右下角的確定按鈕
image-20221022114723561
  • 現在JProfiler已經在監控quarkus應用的進程了,界面如下
image-20221022120025891
  • 如下圖,點擊線程歷史菜單,就能看到當前應用進程內的所有線程,註意按照步驟2過濾一下,只看存活的線程
image-20221022160249366
  • 接下來,咱們就要用JProfiler來觀察常規線程和虛擬線程的區別了
  • 先回憶一下,前文中,咱們開發的quarkus應用有兩個web服務類,分別是:
  1. VTPersonResource.java,該服務類使用了虛擬線程來執行web響應,對應web路徑:/vt/persons

  2. PoolPersonResource.java,該服務類未使用虛擬線程,所以執行web響應的是傳統線程池中的子線程,對應web路徑:/pool/persons

  • 接下來,壓測工具k6先後壓測上述兩個介面,用JProfiler觀察進程中線程的變化情況

不使用虛擬線程時的線程狀況

  • 咱們先發請求到/pool/persons,也就是先不用虛擬線程,看看傳統線程池響應web服務的時候,在JProfiler中是啥樣的
  • 《上篇》那樣,用K6壓測介面/pool/persons,腳本如下,註意IP地址不能用localhost,因為這是在docker容器內運行的,localhost代表容器的迴環網卡,而並非宿主機的:
import http from 'k6/http';
import { sleep, check } from 'k6';

export let options = {
  vus: 10,
  duration: '60s',
};

export default function () {
  let r = Math.floor(Math.random() * 6) + 1;
  const res = http.get(`http://192.168.3.187:8080/pool/persons/${r}`);
  check(res, {
    'is status 200': (res) => res.status === 200,
    'body size is > 0': (r) => r.body.length > 0,
  });
  sleep(1);
}
  • 在壓測期間去看JProfiler,如下圖紅框,新增了10個線程,它們就是負責處理web響應的線程(前文的實戰中,我們已見過web響應的內容,裡面就有線程名稱,紅框中的和它們一致)

image-20221022160442914

  • 下圖是K6的測試報告,可見一共發起了570次請求,然而壓測期間JProfiler上新增的線程只有上圖中的十個,這也印證了線程池的邏輯:每個線程執行完業務邏輯後,回到線程池,下一次請求到來時,該線程繼續執行業務邏輯
image-20221022161058526
  • k6壓測結束後,等上三十秒再去看JProfiler,如下圖,那些處理web響應的子線程已經不見了(或者說不是存活狀態了)
image-20221022161339902
  • 如果您熟悉Java的線程池原理,對以上情況就一目瞭然:線程池空閑時,保留線程數不超過corePoolSize
  • 既然看過了傳統線程池的服務情況,接著改看虛擬線程的情況了,兩邊對比著看收穫一定不小

思考:用JProfiler觀察虛擬線程,你到底想收穫什麼?(本篇精華段落)

  • 大家好,接下來這一段話,個人覺得是本篇的精華,因為這是欣宸自己在迷茫中找到方向的一種方法(或者套路),希望能給您帶來參考

  • 在用JProfiler觀察虛擬線程之前,咱們先來捋捋:接下來咱們究竟想看到什麼,能用文字說清楚嗎?這個問題很重要!

  • 僅僅是想看一眼虛擬線程嗎?那無非就是看到幾個新增的線程,名字有些特殊,僅此而已,這能有啥收穫?

  • 不要急於動手,咱們都應該冷靜下來,認真思考,讓這個問題能用文字表達出來,而不是僅僅在心中有個運行JProfiler的衝動:藉助JProfiler,咱們真正想要的是證虛擬線程的來龍去脈,也就是把官方文檔中的理論,在JProfiler中找到實現!

  • 所以,先閱讀虛擬線程的官方文檔吧,放心,咱們只看最關鍵的部分即可,不會涉及長篇大論

  • 打開java官方文檔,找到虛擬線程定義的那段,如下圖,註意紅框中的內容以及我的中文註解(我將下麵這幅圖稱為本篇最有價值的地方)

image-20221022175154277

  • 沒錯,官方文檔雖多,但咱們沒必要全看,上面這段才是關鍵,看完後捋捋流程圖如下
image-20221022212357583
  • 看到上圖,您應該會有以下三個疑惑:
  1. ForkJoin線程池啥時候創建的?會不會銷毀?
  2. 調度器(scheduler)啥時候創建的?會不會銷毀?
  3. carrier啥時候創建的?會不會銷毀?
  • 如果這些關鍵問題沒說清楚,上面的流程圖算不算是捋了個寂寞...
  • 要想搞清楚為什麼沒有回答上面三個問題,咱們把官方文檔滾動到最頂部,如下圖
image-20221022213416135
  • 上圖紅框表明,這是一篇JEP文檔,即: JDK Enhancement Proposal ,這類文檔只提出標準,而非實現,真正實現的這個標準的,是各個JVM虛擬機廠家(例如Oracle),所以,要想回答上面三個問題,只能去查找具體JDK軟體的實現
  • 簡單的說:別糾結那三個問題,我答不上來...
  • 咱們繼續,接下來更精彩
  • 看過官方資料後,再回到最初的問題,咱們想通過JProfiler得到什麼?相信您已經很清楚了吧,我覺得是這三樣:
  1. 調度器,scheduler(ForkJoin線程池中的線程)
  2. 執行虛擬線程任務的真實線程,carrier
  3. 虛擬線程
  • 現在開始壓測吧,繼續用k6,如下圖,腳本中的地址要改成使用虛擬線程的web服務
image-20221022171002446
  • 壓測期間去觀察JProfiler,如下圖,完全符合預期,說實話,第一次看到這些內容時,自己的內心是很激動的,這種知識點得到印證的感覺真是太好了

image-20221022215116960

  • 再看看那些不再存活的線程,如下圖,大量VirtualThreads存在,這也符合虛擬線程的特性:不復用,執行完畢就結束

image-20221022220055370

  • 等到壓測結束後,scheduler、carrier、虛擬線程,它們都不再存活,如下圖
image-20221022221215621
  • 如此看來,在執行任務的時候,會出現sheduler和carrier來完成虛擬線程中的任務,等到這些任務執行完畢,所有真實線程、虛擬線程都被結束,不再存活
  • 至此,藉助JProfiler觀察常規線程和虛擬線程的實戰就完成了,經過了這些理論結合實際的操作和分析,相信您對虛擬線程的認知已經更具體和全面,如今它不再神秘或者高深莫測,咱們也更有信心學好它用好它

我有個想法

  • 碼字碼到這裡,我想拋出一個大膽的想法和大家一起討論:今天咱們藉助JProfiler觀察到了scheduler、carrier、虛擬線程等的創建、運行、結束等過程,我這裡用的虛擬機是azul JDK,所以JProfiler中看到的也只是azul JDK對虛擬線程規範的實現情況,如果換成其他JDK,例如Oracle JDK,那麼在JProfiler中看到的scheduler、carrier、虛擬線程它們會不會有所不同呢?(例如scheduler可能會存活得久一些)畢竟JEP 425只是個標準,沒有明確規定實現,而azul JDK和Oracle JDK屬於不同廠商的實現
  • 當然了這隻是個猜測,篇(lan)幅(de)所(dong)限(shou)就不在本篇做這些事情了,當我相信會有愛動手的讀者去實戰操作的,麻煩您告訴欣宸一下您的驗證結果,謝謝啦!
  • 寫到這裡,虛擬線程的文章可以完結了嗎?不會,接下來咱們還要暢游quarkus,揭秘@RunOnVirtualThread註解背後的故事,看看優秀的框架是如何玩轉虛擬線程的,上廣告詞:欣宸原創,不辜負您的期待!

歡迎關註博客園:程式員欣宸

學習路上,你不孤單,欣宸原創一路相伴...


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 一、背景介紹 我們團隊一直在持續推進業務系統的體系化治理工作,在這個過程中我們沉澱了自己的DDD腳手架項目。腳手架項目是體系化治理過程中比較重要的一環,它的作用有兩點: 可以對新建的項目進行統一的規範 對於指導老項目進行DDD的改造提供指導 本文主要是梳理和總結了DDD腳手架使用中的編碼規範以及遇到 ...
  • 在設備維修保養管理中,及時通知相關人員是確保設備得到及時維護的關鍵。API介面提供了一個方便的方式來自動發送維修保養通知,以確保工作流程的順利進行。本文將詳細介紹如何使用成熟的API介面來發送設備維修保養通知,以確保設備得到及時的維護,同時提供通俗易懂的步驟和代碼說明。 什麼是API介面? 首先,讓 ...
  • Python安裝 許多PC和Mac已經預裝了Python。 要檢查在Windows PC上是否安裝了Python,請在開始菜單中搜索Python,或在命令行(cmd.exe)上運行以下命令: C:\Users\Your Name>python --version 要檢查在Linux或Mac上是否安裝 ...
  • 原文在這裡。 由 Johan Brandhorst-Satzkorn, Julien Fabre, Damian Gryski, Evan Phoenix, and Achille Roussel 發佈於 2023年9月13日 Go 1.21添加了一個新的埠,通過新的GOOS值wasip1來定位W ...
  • Matplotlib庫 由於誕生的比較早,所以其預設的顯示樣式很難符合現在的審美,這也是它經常為人詬病的地方。 不過,經過版本更迭之後,現在 Matplotlib 已經內置了很多樣式表,通過使用不同的樣式表,可以整體改變繪製圖形的風格,不用再調整一個個顯示參數。 1. 樣式表的使用 1.1. 所有內 ...
  • 在日常開發中我們難免會遇到各種各樣的異常問題的發生,但是任何異常如果都在異常可能會出現的地方直接去處理會讓業務邏輯顯得很臃腫,代碼看上去很冗餘。在開發Web應用程式時,異常處理是一項非常重要的任務。異常處理可以提高程式的健壯性和穩定性。Java後端開發人員可以設計一個統一的全局異常處理方案來解決異常 ...
  • 什麼是minio 引用官網: MinIO是根據GNU Affero通用公共許可證v3.0發佈的高性能對象存儲。它與Amazon S3雲存儲服務相容。使用MinIO構建用於機器學習,分析和應用程式數據工作負載的高性能基礎架構。 官網地址: https://min.io/ 文檔地址: https://d ...
  • 本文通過多個SpringBoot實際項目進行歸納整理,從統一介面返回結果和配置全局異常處理兩個方面出發,介紹如何優雅的封裝規範後端介面輸出,詳細刨析@RestControllerAdvice和@ExceptionHandler註解及使用方式,增加後端服務健壯性和與前端對接規範性 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...