欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java8 如何正確高效的使用并行流

 更新時(shí)間:2021年11月04日 16:11:38   作者:小小工匠  
這篇文章主要介紹了Java8 如何正確高效的使用并行流,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教

正確使用并行流,避免共享可變狀態(tài)

錯(cuò)用并行流而產(chǎn)生錯(cuò)誤的首要原因,就是使用的算法改變了某些共享狀態(tài)。下面是另一種實(shí)現(xiàn)對(duì)前n個(gè)自然數(shù)求和的方法,但這會(huì)改變一個(gè)共享累加器:

public static long sideEffectSum(long n) {
	Accumulator accumulator = new Accumulator();
	LongStream.rangeClosed(1, n).forEach(accumulator::add);
	return accumulator.total;
}
public class Accumulator {
	public long total = 0;
	public void add(long value) { total += value; }
}

有什么問題呢?

它在本質(zhì)上就是順序的。每次訪問 total 都會(huì)出現(xiàn)數(shù)據(jù)競爭。如果用同步來修復(fù),那就完全失去并行的意義了。

為了說明這一點(diǎn),讓我們?cè)囍?Stream 變成并行的:

public static long sideEffectParallelSum(long n) {
	Accumulator accumulator = new Accumulator();
	LongStream.rangeClosed(1, n).parallel().forEach(accumulator::add);
	return accumulator.total;
}

測試下,輸出

在這里插入圖片描述

在這里插入圖片描述

性能無關(guān)緊要了,唯一要緊的是每次執(zhí)行都會(huì)返回不同的結(jié)果,都離正確值差很遠(yuǎn)。這是由于多個(gè)線程在同時(shí)訪問累加器,執(zhí)行 total += value ,而這卻不是一個(gè)原子操作。問題的根源在于, forEach 中調(diào)用的方法有副作用它會(huì)改變多個(gè)線程共享的對(duì)象的可變狀態(tài)。

要是你想用并行 Stream 又不想引發(fā)類似的意外,就必須避免這種情況。

所以共享可變狀態(tài)會(huì)影響并行流以及并行計(jì)算,要避免共享可變狀態(tài),確保并行 Stream 得到正確的結(jié)果。

高效使用并行流

是否有必要使用并行流?

  • 如果有疑問,多次測試結(jié)果。把順序流轉(zhuǎn)成并行流輕而易舉,但卻不一定是好事
  • 留意裝箱。自動(dòng)裝箱和拆箱操作會(huì)大大降低性能

Java 8中有原始類型流( IntStream 、LongStream 、 DoubleStream )來避免這種操作,但?有可能都應(yīng)該用這些流。

  • 有些操作本身在并行流上的性能就比順序流差。特別是 limit 和 findFirst 等依賴于元素順序的操作,它們?cè)诓⑿辛魃蠄?zhí)行的代價(jià)非常大。

例如, findAny 會(huì)比 findFirst 性能好,因?yàn)樗灰欢ㄒ错樞騺韴?zhí)行??梢哉{(diào)用 unordered 方法來把有序流變成無序流。那么,如果你需要流中的n個(gè)元素而不是專門要前n個(gè)的話,對(duì)無序并行流調(diào)用limit 可能會(huì)比單個(gè)有序流(比如數(shù)據(jù)源是一個(gè) List )更高效。

  • 還要考慮流的操作流水線的總計(jì)算成本。

設(shè)N是要處理的元素的總數(shù),Q是一個(gè)元素通過流水線的大致處理成本,則N*Q就是這個(gè)對(duì)成本的一個(gè)粗略的定性估計(jì)。Q值較高就意味著使用并行流時(shí)性能好的可能性比較大。

  • 對(duì)于較小的數(shù)據(jù)量,選擇并行流幾乎從來都不是一個(gè)好的決定。并行處理少數(shù)幾個(gè)元素的好處還?不上并行化造成的額外開銷
  • 要考慮流背后的數(shù)據(jù)結(jié)構(gòu)是否易于分解。

例如, ArrayList 的拆分效率比 LinkedList高得多,因?yàn)榍罢哂貌恢闅v就可以平均拆分,而后者則必須遍歷。另外,用 range 工廠方法創(chuàng)建的原始類型流也可以快速分解。

  • 流自身的特點(diǎn),以及流水線中的中間操作修改流的方式,都可能會(huì)改變分解過程的性能。

例如,一個(gè) SIZED 流可以分成大小相等的兩部分,這樣每個(gè)部分都可以比較高效地并行處理,但篩選操作可能丟棄的元素個(gè)數(shù)卻無法預(yù)測,導(dǎo)致流本身的大小未知。

  • 還要考慮終端操作中合并步驟的代價(jià)是大是小(例如 Collector 中的 combiner 方法)

如果這一步代價(jià)很大,那么組合每個(gè)子流產(chǎn)生的部分結(jié)果所付出的代價(jià)就可能會(huì)超出通過并行流得到的性能提升。

流的數(shù)據(jù)源和可分解性

在這里插入圖片描述

最后, 并行流背后使用的基礎(chǔ)架構(gòu)是Java 7中引入的分支/合并框架了解它的內(nèi)部原理至關(guān)重要。

java 并行計(jì)算的幾點(diǎn)實(shí)踐總結(jié)

稍微接觸了 java 的并行計(jì)算,談?wù)剮c(diǎn)淺顯的總結(jié)吧

并行計(jì)算不一定比串行計(jì)算快,一般在大規(guī)模問題才會(huì)顯示出優(yōu)勢

結(jié)合 lambda 表達(dá)式的 parallelStream 可以方便調(diào)用并行計(jì)算,但可能會(huì)出現(xiàn)空指針錯(cuò)誤,解決這一問題可能需要更高級(jí)的多線程知識(shí)

看網(wǎng)上資料,Collection 類型對(duì)并行計(jì)算支持的好,一般數(shù)組類型支持的一般。

以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。

相關(guān)文章

  • 解決eclipse上傳svn忽略target文件夾的坑

    解決eclipse上傳svn忽略target文件夾的坑

    這篇文章主要介紹了解決eclipse上傳svn忽略target文件夾的坑,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧
    2021-01-01
  • JAVA多線程處理for循環(huán)數(shù)據(jù)詳細(xì)講解

    JAVA多線程處理for循環(huán)數(shù)據(jù)詳細(xì)講解

    這篇文章主要給大家介紹了關(guān)于JAVA多線程處理for循環(huán)數(shù)據(jù)的相關(guān)資料,我們?cè)诖a中經(jīng)常需要使用for循環(huán)這個(gè)操作來達(dá)到目的,而當(dāng)for循環(huán)的次數(shù)過多時(shí)我們會(huì)發(fā)現(xiàn)執(zhí)行效率會(huì)變的很低,整體耗時(shí)非常多,需要的朋友可以參考下
    2023-07-07
  • java多線程編程實(shí)現(xiàn)下雪效果

    java多線程編程實(shí)現(xiàn)下雪效果

    這篇文章主要介紹了java多線程編程實(shí)現(xiàn)下雪效果的相關(guān)資料,需要的朋友可以參考下
    2015-11-11
  • springcloud項(xiàng)目里application.yml不加載的坑及解決

    springcloud項(xiàng)目里application.yml不加載的坑及解決

    這篇文章主要介紹了springcloud項(xiàng)目里application.yml不加載的坑及解決,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2024-07-07
  • SpringCloud zookeeper作為注冊(cè)中心使用介紹

    SpringCloud zookeeper作為注冊(cè)中心使用介紹

    ZooKeeper由雅虎研究院開發(fā),是Google Chubby的開源實(shí)現(xiàn),后來托管到Apache,于2010年11月正式成為Apache的頂級(jí)項(xiàng)目。ZooKeeper是一個(gè)經(jīng)典的分布式數(shù)據(jù)一致性解決方案,致力于為分布式應(yīng)用提供一個(gè)高性能、高可用,且具有嚴(yán)格順序訪問控制能力的分布式協(xié)調(diào)服務(wù)
    2022-11-11
  • springboot本地調(diào)試沒問題,打包運(yùn)行報(bào)錯(cuò)原因及分析

    springboot本地調(diào)試沒問題,打包運(yùn)行報(bào)錯(cuò)原因及分析

    這篇文章主要介紹了springboot本地調(diào)試沒問題,打包運(yùn)行報(bào)錯(cuò)原因及分析,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-05-05
  • oracle+mybatis-plus+springboot實(shí)現(xiàn)分頁查詢的實(shí)例

    oracle+mybatis-plus+springboot實(shí)現(xiàn)分頁查詢的實(shí)例

    本文主要介紹了oracle+mybatis-plus+springboot實(shí)現(xiàn)分頁查詢,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2021-08-08
  • Java泛型的概念、定義與運(yùn)行方法分析

    Java泛型的概念、定義與運(yùn)行方法分析

    這篇文章主要介紹了Java泛型的概念、定義與運(yùn)行方法,結(jié)合實(shí)例形式分析了java泛型的基本概念、定義與IDEA、命令行兩種運(yùn)行方法,需要的朋友可以參考下
    2019-08-08
  • Java 注解學(xué)習(xí)筆記

    Java 注解學(xué)習(xí)筆記

    一直都在使用注解,但是一直都沒有用的很明白,后來被逼的發(fā)現(xiàn)不搞明白真的就沒有辦法愉快的寫代碼了,所以,這篇《Java中的注解學(xué)習(xí)筆記》就呼之欲出了
    2020-10-10
  • 詳解Java的Proxy動(dòng)態(tài)代理機(jī)制

    詳解Java的Proxy動(dòng)態(tài)代理機(jī)制

    Java有兩種代理方式,一種是靜態(tài)代理,另一種是動(dòng)態(tài)代理。對(duì)于靜態(tài)代理,其實(shí)就是通過依賴注入,對(duì)對(duì)象進(jìn)行封裝,不讓外部知道實(shí)現(xiàn)的細(xì)節(jié)。很多 API 就是通過這種形式來封裝的
    2021-06-06

最新評(píng)論