」工欲善其事,必先利其器。「—孔子《論語.錄靈公》
首頁 > 程式設計 > 探索 JVM 虛擬執行緒機制中的固定

探索 JVM 虛擬執行緒機制中的固定

發佈於2024-11-08
瀏覽:676

Java 的虚拟线程提供了传统操作系统线程的轻量级替代方案,从而实现了高效的并发管理。但了解他们的行为对于获得最佳表现至关重要。这篇博文深入探讨了固定(一种可能影响虚拟线程执行的场景),并探讨了监控和解决该问题的技术。

虚拟线程:一种轻量级并发方法

Java 的虚拟线程是运行在底层操作系统线程(载体线程)之上的托管实体。与创建大量操作系统线程相比,它们提供了一种更有效的处理并发的方法,因为它们产生的开销较低。 JVM动态地将虚拟线程映射到承载线程,从而更好地利用资源。

  • 由 JVM 管理:与操作系统直接管理的 OS 线程不同,虚拟线程由 Java 虚拟机 (JVM) 创建和调度。这允许在 JVM 环境中进行更细粒度的控制和优化。

  • 减少开销:与操作系统线程相比,创建和管理虚拟线程所产生的开销显着降低。这是因为 JVM 可以利用较少数量的底层操作系统线程来有效管理更大的虚拟线程池。

  • 与现有代码的兼容性:虚拟线程被设计为与现有Java代码无缝集成。它们可以与传统操作系统线程一起使用,并在 Executor 和 ExecutorService 等熟悉的结构中工作,用于管理并发。

下图展示了虚拟线程和平台线程的关系:

Exploring Pinning in JVM


固定:当虚拟线程卡住时

当虚拟线程与其承载线程绑定时,就会发生固定。这本质上意味着虚拟线程在处于固定状态时不能被抢占(切换到另一个承载线程)。以下是触发固定的常见场景:

  • 同步块和方法:在同步块或方法中执行代码会导致固定。这确保了对共享资源的独占访问,防止数据损坏问题。

代码示例:

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

public class Main {

  public static void main(String[] args) throws InterruptedException {

    final Counter counter = new Counter();

    Runnable task = () -> {
      for (int i = 0; i 



在此示例中,当虚拟线程进入同步块时,它会固定到其承载线程,但这并不总是正确的。仅Java的synchronized关键字不足以导致虚拟线程中的线程固定。为了发生线程固定,同步块内必须有一个阻塞点,该阻塞点会导致虚拟线程触发暂停,并最终不允许从其承载线程卸载。线程固定可能会导致性能下降,因为它会抵消使用轻量级/虚拟线程的好处。

每当虚拟线程遇到阻塞点时,其状态就会转换为 PARKING。这种状态转换是通过调用 VirtualThread.park() 方法来指示的:

// JDK core code
void park() {
  assert Thread.currentThread() == this;
  // complete immediately if parking permit available or interrupted
  if (getAndSetParkPermit(false) || interrupted)
    return;
  // park the thread
  setState(PARKING);
  try {
    if (!yieldContinuation()) {
      // park on the carrier thread when pinned
      parkOnCarrierThread(false, 0);
    }
  } finally {
    assert (Thread.currentThread() == this) && (state() == RUNNING);
  }
}

让我们看一个代码示例来说明这个概念:

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

public class Main {

  public static void main(String[] args) {

    Counter counter = new Counter();

    Runnable task = () -> {
      for (int i = 0; i 



  • 本机方法/外部函数:运行本机方法或外部函数也可能导致固定。在这些操作期间,JVM 可能无法有效管理虚拟线程的状态。

使用 -Djdk.tracePinnedThreads=full 监视固定

-Djdk.tracePinnedThreads=full 标志是一个 JVM 启动参数,它提供有关虚拟线程固定的详细跟踪信息。启用后,它会记录以下事件:

  • 固定涉及的虚拟线程ID
  • 虚拟线程固定的承载线程ID
  • 堆栈跟踪指示导致固定的代码部分

仅在调试会话期间明智地使用此标志,因为它会带来性能开销。

  1. 编译我们的演示代码:

    javac Main.java
    
  2. 使用 -Djdk.tracePinnedThreads=full 标志启动编译的代码:

    java -Djdk.tracePinnedThreads=full Main
    
  3. 观察控制台中的输出,其中显示了有关虚拟线程固定的详细信息:

    Thread[#29,ForkJoinPool-1-worker-1,5,CarrierThreads]
    java.base/java.lang.VirtualThread$VThreadContinuation.onPinned(VirtualThread.java:183)
    java.base/jdk.internal.vm.Continuation.onPinned0(Continuation.java:393)
    java.base/java.lang.VirtualThread.parkNanos(VirtualThread.java:621)
    java.base/java.lang.VirtualThread.sleepNanos(VirtualThread.java:791)
    java.base/java.lang.Thread.sleep(Thread.java:507)
    Counter.increment(Main.java:38) 
    
    

使用可重入锁修复固定

固定是一种不良情况,它会影响虚拟线程的性能。可重入锁是抵消固定的有效工具。以下是如何使用可重入锁来缓解锁定情况:

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.locks.ReentrantLock;

public class Main {

  public static void main(String[] args) {

    Counter counter = new Counter();

    Runnable task = () -> {
      for (int i = 0; i 



在更新的示例中,我们使用 ReentrantLock 而不是同步块。线程可以获取锁并在完成操作后立即释放它,与可能持有锁较长时间的同步块相比,可能会减少锁定的持续时间。

综上所述

Java 的虚拟线程证明了该语言的发展和功能。它们为传统操作系统线程提供了一种全新的、轻量级的替代方案,为高效的并发管理提供了一座桥梁。花时间深入挖掘并理解线程固定等关键概念可以让开发人员掌握充分利用这些轻量级线程潜力的专业知识。这些知识不仅帮助开发人员为利用即将推出的功能做好准备,还使他们能够在当前项目中更有效地解决复杂的并发控制问题。

版本聲明 本文轉載於:https://dev.to/yanev/exploring-pinning-in-jvms-virtual-thread-mechanism-5h13?1如有侵犯,請聯絡[email protected]刪除
最新教學 更多>
  • 編譯器報錯“usr/bin/ld: cannot find -l”解決方法
    編譯器報錯“usr/bin/ld: cannot find -l”解決方法
    錯誤:“ usr/bin/ld:找不到-l “ 此錯誤表明鏈接器在鏈接您的可執行文件時無法找到指定的庫。為了解決此問題,我們將深入研究如何指定庫路徑並將鏈接引導到正確位置的詳細信息。 添加庫搜索路徑的一個可能的原因是,此錯誤是您的makefile中缺少庫搜索路徑。要解決它,您可以在鏈接器命令中添...
    程式設計 發佈於2025-04-30
  • JavaScript數組創建與操作技巧
    JavaScript數組創建與操作技巧
    深入浅出JavaScript数组:创建与操作详解 核心要点 JavaScript数组拥有length属性,可进行操作,并具有编号属性,名称范围在0到4294967294(含)之间。JavaScript不支持关联数组。 JavaScript数组的创建方式多样,建议使用数组字面量创建新数组。数组可以是密...
    程式設計 發佈於2025-04-30
  • 圖片在Chrome中為何仍有邊框? `border: none;`無效解決方案
    圖片在Chrome中為何仍有邊框? `border: none;`無效解決方案
    在chrome 中刪除一個頻繁的問題時,在與Chrome and IE9中的圖像一起工作時,遇到了一個頻繁的問題。和“邊境:無;”在CSS中。要解決此問題,請考慮以下方法: Chrome具有忽略“ border:none; none;”的已知錯誤,風格。要解決此問題,請使用以下CSS ID塊創建帶...
    程式設計 發佈於2025-04-30
  • 如何有效地選擇熊貓數據框中的列?
    如何有效地選擇熊貓數據框中的列?
    在處理數據操作任務時,在Pandas DataFrames 中選擇列時,選擇特定列的必要條件是必要的。在Pandas中,選擇列的各種選項。 選項1:使用列名 如果已知列索引,請使用ILOC函數選擇它們。請注意,python索引基於零。 df1 = df.iloc [:,0:2]#使用索引0和1 ...
    程式設計 發佈於2025-04-30
  • 如何使用“ JSON”軟件包解析JSON陣列?
    如何使用“ JSON”軟件包解析JSON陣列?
    parsing JSON與JSON軟件包 QUALDALS:考慮以下go代碼:字符串 } func main(){ datajson:=`[“ 1”,“ 2”,“ 3”]`` arr:= jsontype {} 摘要:= = json.unmarshal([] byte(...
    程式設計 發佈於2025-04-30
  • 為什麼不該用%v打印整數和字符串?
    為什麼不該用%v打印整數和字符串?
    將%v用於打印整數和字符串 的後果,雖然可以使用%v都打印整數和字符串,但不是推薦的方法。使用%v用於整數可能會導致格式不一致,因為默認格式可能會根據整數的值而改變。例如,大整數可以用逗號作為分離器進行格式化,而在沒有分離器的情況下可以打印小整數。 使用%v用於字符串也可能導致意外的行為。默認情況...
    程式設計 發佈於2025-04-30
  • 如何從Python中的字符串中刪除表情符號:固定常見錯誤的初學者指南?
    如何從Python中的字符串中刪除表情符號:固定常見錯誤的初學者指南?
    從python import codecs import codecs import codecs 導入 text = codecs.decode('這狗\ u0001f602'.encode('utf-8'),'utf-8') 印刷(文字)#帶有...
    程式設計 發佈於2025-04-30
  • 如何使用不同數量列的聯合數據庫表?
    如何使用不同數量列的聯合數據庫表?
    合併列數不同的表 當嘗試合併列數不同的數據庫表時,可能會遇到挑戰。一種直接的方法是在列數較少的表中,為缺失的列追加空值。 例如,考慮兩個表,表 A 和表 B,其中表 A 的列數多於表 B。為了合併這些表,同時處理表 B 中缺失的列,請按照以下步驟操作: 確定表 B 中缺失的列,並將它們添加到表的...
    程式設計 發佈於2025-04-30
  • 為什麼HTML無法打印頁碼及解決方案
    為什麼HTML無法打印頁碼及解決方案
    無法在html頁面上打印頁碼? @page規則在@Media內部和外部都無濟於事。 HTML:Customization:@page { margin: 10%; @top-center { font-family: sans-serif; font-weight: ...
    程式設計 發佈於2025-04-30
  • C++20 Consteval函數中模板參數能否依賴於函數參數?
    C++20 Consteval函數中模板參數能否依賴於函數參數?
    [ consteval函數和模板參數依賴於函數參數在C 17中,模板參數不能依賴一個函數參數,因為編譯器仍然需要對非contexexpr futcoriations contim at contexpr function進行評估。 compile time。 C 20引入恆定函數,必須在編譯時進...
    程式設計 發佈於2025-04-30
  • 如何避免AngularJS中因URL無效導致的背景圖錯誤?
    如何避免AngularJS中因URL無效導致的背景圖錯誤?
    的背景圖像錯誤在AngularJS中使用無效的URL在AngularJS中的URL中錯誤,NG-SRC標籤可確保具有動態變量的URL在Angular評估它們之前不會引起錯誤。但是,當使用背景圖像設置背景圖像時,通常會發生類似的錯誤:url(...)。 發生這種情況,因為Angular不會評估CSS...
    程式設計 發佈於2025-04-30
  • C#整數轉二進制高效方法
    C#整數轉二進制高效方法
    C# 中整數到二進製表示的轉換 將整數轉換為其二進製表示是常見的編程任務。在 C# 中,有多種方法可以執行此轉換,包括使用 Convert 類的 ToInt32 和 ToString 方法。 為了演示此過程,讓我們解決一個用戶遇到的問題,該用戶嘗試將表示為字符串的整數轉換為其二進製表示: Str...
    程式設計 發佈於2025-04-30
  • 為什麼我在Silverlight Linq查詢中獲得“無法找到查詢模式的實現”錯誤?
    為什麼我在Silverlight Linq查詢中獲得“無法找到查詢模式的實現”錯誤?
    查詢模式實現缺失:解決“無法找到”錯誤在銀光應用程序中,嘗試使用LINQ建立錯誤的數據庫連接的嘗試,無法找到以查詢模式的實現。 ”當省略LINQ名稱空間或查詢類型缺少IEnumerable 實現時,通常會發生此錯誤。 解決問題來驗證該類型的質量是至關重要的。在此特定實例中,tblpersoon可能...
    程式設計 發佈於2025-04-30
  • 在Java中使用for-to-loop和迭代器進行收集遍歷之間是否存在性能差異?
    在Java中使用for-to-loop和迭代器進行收集遍歷之間是否存在性能差異?
    For Each Loop vs. Iterator: Efficiency in Collection TraversalIntroductionWhen traversing a collection in Java, the choice arises between using a for-...
    程式設計 發佈於2025-04-30
  • 如何在鼠標單擊時編程選擇DIV中的所有文本?
    如何在鼠標單擊時編程選擇DIV中的所有文本?
    在鼠標上選擇div文本單擊帶有文本內容,用戶如何使用單個鼠標單擊單擊div中的整個文本?這允許用戶輕鬆拖放所選的文本或直接複製它。 在單個鼠標上單擊的div元素中選擇文本,您可以使用以下Javascript函數: function selecttext(canduterid){ if(d...
    程式設計 發佈於2025-04-30

免責聲明: 提供的所有資源部分來自互聯網,如果有侵犯您的版權或其他權益,請說明詳細緣由並提供版權或權益證明然後發到郵箱:[email protected] 我們會在第一時間內為您處理。

Copyright© 2022 湘ICP备2022001581号-3