中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Java8與Runtime.getRuntime().availableProcessors()的詳細解析

發布時間:2020-07-22 09:09:01 來源:億速云 閱讀:1471 作者:小豬 欄目:編程語言

這篇文章主要講解了Java8與Runtime.getRuntime().availableProcessors()的詳細解析,內容清晰明了,對此有興趣的小伙伴可以學習一下,相信大家閱讀完之后會有幫助。

lambda表達式以及并行流。官方承諾你寫出來的代碼更運行得更快。流會自動通過Fork/Join池并行地執行。我聽過一些關于Java 8的主題的演講,不過在這個非常關鍵的點上它們都說的有點問題。我計劃在后續的文章中對并行流進行下深入的講解,在這之前我先花點時間仔細地分析下它。關于這個問題,我只想問你們一個非常簡單的問題,不過也是一個非常重要的問題,因為它是很多問題的關鍵所在。這個問題是:

這些并行操作的線程都是從哪來的?

在Java 8里,我們有一個通用的Fork/Join池,我們可以通過ForkJoinPool.commonPool()來訪問它。并行流,并行排序,CompletableFuture等都會用到它。當你構造一個Fork/Join池的時候,通常你都沒有指定最大線程數。你只是指定了一個期望的并發數,也就是說你希望在運行時的同一時間有多少活躍的線程。當線程被阻塞在一個phaser的時候,會創建另一個線程來保證池里有足夠的活躍線程。這個phaser就是觸發這個行為的同步器。Fork/Join池最大的線程數是32767,但在遠沒達到這個數量時,在大多數操作系統上就會拋出OutOfMemoryError異常了。在這段示例代碼中,我會不斷創建新的RecursiveAction真到達到第一個階段(也就是到達了200個線程)。如果我們增加到一個更大的數字,比如說到100000,這段代碼就會失敗了。

import java.util.concurrent.*;

public class PhaserForkJoin {
 public static void main(String... args) {
  ForkJoinPool common = ForkJoinPool.commonPool();
  Phaser phaser = new Phaser(200);
  common.invoke(new PhaserWaiter(phaser));
 }

 private static class PhaserWaiter extends RecursiveAction {
  private final Phaser phaser;

  private PhaserWaiter(Phaser phaser) {
   this.phaser = phaser;
   System.out.println(ForkJoinPool.commonPool().getPoolSize());
  }

  protected void compute() {
   if (phaser.getPhase() > 0) return; // we've passed first phase
   PhaserWaiter p1 = new PhaserWaiter(phaser);
   p1.fork();
   phaser.arriveAndAwaitAdvance();
   p1.join();
  }
 }
}

Fork/Join池沒有一個最大線程數,只有一個期望并發數,這是指我們希望同時有多少個活躍線程。

通用池是很有用的,因為它意味著不同類型的作業可以共享同一個池,而不用超出代碼所運行的機器上期望并發數。當然了,如果一個線程由于非Phaser的其它原因阻塞了,那可能這個通用池的表現就和預期的不太一樣了。

什么是通用FJ池的默認的期望并發數?

通常的FJ池的期望并發數的默認值是Runtime.getRuntime().availableProcessors() -1。如果你在一個雙核的機器上通過Arrays.parallelSort()來運行并行排序的話,默認使用的是普通的Arrays.sort()方法。盡管Oracle的官方文檔可能許諾你可以獲得性能提升,但是你在一個雙核的機器上可能完全看不著任何提升。

然而,更大的問題在于Runtime.getRuntime().availableProcessors()也并非都能返回你所期望的數值。比如說,在我的雙核1-2-1機器上,它返回的是2,這是對的。不過在我的1-4-2機器 上,也就是一個CPU插槽,4核,每個核2個超線程,這樣的話會返回8。不過我其實只有4個核,如果代碼的瓶頸是在CPU這塊的話,我會有7個線程在同時 競爭CPU周期,而不是更合理的4個線程。如果我的瓶頸是在內存這的話,那這個測試我可以獲得7倍的性能提升。

不過這還沒完!Java Champions上的一個哥們發現了一種情況,他有一臺16-4-2的機器 (也就是16個CPU插槽,每個CPU4個核,每核兩個超線程,返回的值居然是16!從我的i7 Macbook pro上的結果來看,我覺得應該返回的是16*4*2=128。在這臺機器上運行Java 8的話,它只會將通用的FJ池的并發數設置成15。正如 Brian Goetz所指出的,“虛擬機其實不清楚什么是處理器,它只是去請求操作系統返回一個值。同樣的,操作系統也不知道怎么回事,它是去問的硬件設備。硬件會告訴它一個值,通常來說是硬件線程數。操作系統相信硬件說的,而虛擬機又相信操作系統說的。”

所幸的是還有一個解決方案。啟動的時候,你可以通過系統屬性 java.util.concurrent.ForkJoinPool.common.parallelism來設置通用池的并發數。也就是說,我們可以通過-Djava.util.concurrent.ForkJoinPool.common.parallelism=128來啟動這段程序,現在你可以看到它的并發數是128了:

import java.util.concurrent.*;

public class ForkJoinPoolCommon {
 public static void main(String... args) {
  System.out.println(ForkJoinPool.commonPool());
 }
}

還有兩個控制通用池的額外的系統屬性。如果你希望處理未捕獲異常的話,你可以通過java.util.concurrent.ForkJoinPool.common.exceptionHandler來指定一個處理類。如果你希望有自己的線程工廠的話,可以通過 java.util.concurrent.ForkJoinPool.common.threadFactory來配置。默認的Fork/Join池的工廠生成的是守護線程,可能你的應用里面不希望使用它。不過如果你這么做的話請小心——這樣你就無法關閉這個通用池了。

看完上述內容,是不是對Java8與Runtime.getRuntime().availableProcessors()的詳細解析有進一步的了解,如果還想學習更多內容,歡迎關注億速云行業資訊頻道。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

巢湖市| 云阳县| 大关县| 星座| 林芝县| 皋兰县| 香河县| 荥经县| 来宾市| 庄河市| 青川县| 白城市| 屏山县| 宿州市| 石门县| 天气| 镶黄旗| 新化县| 简阳市| 宜川县| 吕梁市| 鞍山市| 黔西| 新河县| 清流县| 长葛市| 龙门县| 仙游县| 海城市| 安西县| 潮安县| 克什克腾旗| 友谊县| 修文县| 厦门市| 庆阳市| 金寨县| 白水县| 武宁县| 赫章县| 平遥县|