免费观看又色又爽又黄的小说免费_美女福利视频国产片_亚洲欧美精品_美国一级大黄大色毛片

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

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

10年積累的成都做網(wǎng)站、成都網(wǎng)站建設(shè)經(jīng)驗(yàn),可以快速應(yīng)對(duì)客戶對(duì)網(wǎng)站的新想法和需求。提供各種問(wèn)題對(duì)應(yīng)的解決方案。讓選擇我們的客戶得到更好、更有力的網(wǎng)絡(luò)服務(wù)。我雖然不認(rèn)識(shí)你,你也不認(rèn)識(shí)我。但先網(wǎng)站設(shè)計(jì)后付款的網(wǎng)站建設(shè)流程,更有上黨免費(fèi)網(wǎng)站建設(shè)讓你可以放心的選擇與我們合作。

lambda表達(dá)式以及并行流。官方承諾你寫出來(lái)的代碼更運(yùn)行得更快。流會(huì)自動(dòng)通過(guò)Fork/Join池并行地執(zhí)行。我聽(tīng)過(guò)一些關(guān)于Java 8的主題的演講,不過(guò)在這個(gè)非常關(guān)鍵的點(diǎn)上它們都說(shuō)的有點(diǎn)問(wèn)題。我計(jì)劃在后續(xù)的文章中對(duì)并行流進(jìn)行下深入的講解,在這之前我先花點(diǎn)時(shí)間仔細(xì)地分析下它。關(guān)于這個(gè)問(wèn)題,我只想問(wèn)你們一個(gè)非常簡(jiǎn)單的問(wèn)題,不過(guò)也是一個(gè)非常重要的問(wèn)題,因?yàn)樗呛芏鄦?wèn)題的關(guān)鍵所在。這個(gè)問(wèn)題是:

這些并行操作的線程都是從哪來(lái)的?

在Java 8里,我們有一個(gè)通用的Fork/Join池,我們可以通過(guò)ForkJoinPool.commonPool()來(lái)訪問(wèn)它。并行流,并行排序,CompletableFuture等都會(huì)用到它。當(dāng)你構(gòu)造一個(gè)Fork/Join池的時(shí)候,通常你都沒(méi)有指定最大線程數(shù)。你只是指定了一個(gè)期望的并發(fā)數(shù),也就是說(shuō)你希望在運(yùn)行時(shí)的同一時(shí)間有多少活躍的線程。當(dāng)線程被阻塞在一個(gè)phaser的時(shí)候,會(huì)創(chuàng)建另一個(gè)線程來(lái)保證池里有足夠的活躍線程。這個(gè)phaser就是觸發(fā)這個(gè)行為的同步器。Fork/Join池最大的線程數(shù)是32767,但在遠(yuǎn)沒(méi)達(dá)到這個(gè)數(shù)量時(shí),在大多數(shù)操作系統(tǒng)上就會(huì)拋出OutOfMemoryError異常了。在這段示例代碼中,我會(huì)不斷創(chuàng)建新的RecursiveAction真到達(dá)到第一個(gè)階段(也就是到達(dá)了200個(gè)線程)。如果我們?cè)黾拥揭粋€(gè)更大的數(shù)字,比如說(shuō)到100000,這段代碼就會(huì)失敗了。

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池沒(méi)有一個(gè)最大線程數(shù),只有一個(gè)期望并發(fā)數(shù),這是指我們希望同時(shí)有多少個(gè)活躍線程。

通用池是很有用的,因?yàn)樗馕吨煌愋偷淖鳂I(yè)可以共享同一個(gè)池,而不用超出代碼所運(yùn)行的機(jī)器上期望并發(fā)數(shù)。當(dāng)然了,如果一個(gè)線程由于非Phaser的其它原因阻塞了,那可能這個(gè)通用池的表現(xiàn)就和預(yù)期的不太一樣了。

什么是通用FJ池的默認(rèn)的期望并發(fā)數(shù)?

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

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

不過(guò)這還沒(méi)完!Java Champions上的一個(gè)哥們發(fā)現(xiàn)了一種情況,他有一臺(tái)16-4-2的機(jī)器 (也就是16個(gè)CPU插槽,每個(gè)CPU4個(gè)核,每核兩個(gè)超線程,返回的值居然是16!從我的i7 Macbook pro上的結(jié)果來(lái)看,我覺(jué)得應(yīng)該返回的是16*4*2=128。在這臺(tái)機(jī)器上運(yùn)行Java 8的話,它只會(huì)將通用的FJ池的并發(fā)數(shù)設(shè)置成15。正如 Brian Goetz所指出的,“虛擬機(jī)其實(shí)不清楚什么是處理器,它只是去請(qǐng)求操作系統(tǒng)返回一個(gè)值。同樣的,操作系統(tǒng)也不知道怎么回事,它是去問(wèn)的硬件設(shè)備。硬件會(huì)告訴它一個(gè)值,通常來(lái)說(shuō)是硬件線程數(shù)。操作系統(tǒng)相信硬件說(shuō)的,而虛擬機(jī)又相信操作系統(tǒng)說(shuō)的。”

所幸的是還有一個(gè)解決方案。啟動(dòng)的時(shí)候,你可以通過(guò)系統(tǒng)屬性 java.util.concurrent.ForkJoinPool.common.parallelism來(lái)設(shè)置通用池的并發(fā)數(shù)。也就是說(shuō),我們可以通過(guò)-Djava.util.concurrent.ForkJoinPool.common.parallelism=128來(lái)啟動(dòng)這段程序,現(xiàn)在你可以看到它的并發(fā)數(shù)是128了:

import java.util.concurrent.*;

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

還有兩個(gè)控制通用池的額外的系統(tǒng)屬性。如果你希望處理未捕獲異常的話,你可以通過(guò)java.util.concurrent.ForkJoinPool.common.exceptionHandler來(lái)指定一個(gè)處理類。如果你希望有自己的線程工廠的話,可以通過(guò) java.util.concurrent.ForkJoinPool.common.threadFactory來(lái)配置。默認(rèn)的Fork/Join池的工廠生成的是守護(hù)線程,可能你的應(yīng)用里面不希望使用它。不過(guò)如果你這么做的話請(qǐng)小心——這樣你就無(wú)法關(guān)閉這個(gè)通用池了。

看完上述內(nèi)容,是不是對(duì)Java8與Runtime.getRuntime().availableProcessors()的詳細(xì)解析有進(jìn)一步的了解,如果還想學(xué)習(xí)更多內(nèi)容,歡迎關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道。

分享文章:Java8與Runtime.getRuntime().availableProcessors()的詳細(xì)解析
本文URL:http://m.newbst.com/article44/isjcee.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供定制開(kāi)發(fā)移動(dòng)網(wǎng)站建設(shè)定制網(wǎng)站網(wǎng)頁(yè)設(shè)計(jì)公司做網(wǎng)站企業(yè)建站

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來(lái)源: 創(chuàng)新互聯(lián)

成都網(wǎng)站建設(shè)