自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

深入理解Stream之原理剖析

開發(fā) 前端
中間操作只會進(jìn)行操作記錄,只有結(jié)束操作才會觸發(fā)實(shí)際的計算,可以理解為懶加載,這也是Stream在操作大對象迭代計算的時候如此高效的原因之一。

今天我們先來聊聊深入理解Stream之原理剖析。

圖片

Stream操作分類

圖片

Stream中的操作可以分為兩大類:中間操作與結(jié)束操作。

中間操作只會進(jìn)行操作記錄,只有結(jié)束操作才會觸發(fā)實(shí)際的計算,可以理解為懶加載,這也是Stream在操作大對象迭代計算的時候如此高效的原因之一。

中間操作分為有狀態(tài)操作與無狀態(tài)操作,無狀態(tài)是指元素的處理不受之前元素的影響,有狀態(tài)是指該操作只有拿到所有元素之后才能繼續(xù)下去。這也比較好理解,比如有狀態(tài)的distinct()去重方法,你說他能不關(guān)心其他值嗎?當(dāng)然不能,他必須拿到所有元素才知道當(dāng)前迭代的元素是否被重復(fù)。

結(jié)束操作可以分為短路與非短路操作,這個應(yīng)該很好理解,短路是指遇到某些符合條件的元素就可以得到最終結(jié)果;而非短路是指必須處理所有元素才能得到最終結(jié)果。

之所以要進(jìn)行如此精細(xì)的劃分,是因?yàn)榈讓訉γ恳环N情況的處理方式不同。

Stream結(jié)構(gòu)分析

讓我們先簡單看看下面一段代碼:

List<String> list = new ArrayList<>();

// 獲取stream1
Stream<String> stream1 = list.stream();

// stream1通過filter后得到stream2
Stream<String> stream2 = stream1.filter("lige"::equals);

// stream1與stream2是同一個對象嗎?
System.out.println("stream1.equals(stream2) = " + stream1.equals(stream2));
System.out.println("stream1.classTypeName = " + stream1.getClass().getTypeName());
System.out.println("stream2.classTypeName = " + stream2.getClass().getTypeName());

// 結(jié)果
// stream1.equals(stream2) = false
// stream1.classTypeName = java.util.stream.ReferencePipeline$Head
// stream1.classTypeName = java.util.stream.ReferencePipeline$2

很明顯,stream1與stream2不是同一個對象,并且他們不是同一個實(shí)現(xiàn)類。stream1的實(shí)現(xiàn)類為ReferencePipeline$Head,而stream2的實(shí)現(xiàn)類為一個匿名內(nèi)部類,讓我們進(jìn)步一分析其源碼,所謂源碼之下,無所遁形。

圖片

圖片

圖片

讓我們再看看stream2:

圖片

圖片

圖片

圖片

圖片

圖片

通過分析我們可以發(fā)現(xiàn),stream2的實(shí)現(xiàn)類是StatelessOp,所以就形成了這樣一個結(jié)構(gòu)。

圖片

每一次中間操作都會生成一個新的Stream,如果是無狀態(tài)操作則實(shí)現(xiàn)類是StatelessOp,如果是有狀態(tài)操作則實(shí)現(xiàn)類是StatefulOp。

讓我們再來看一下他們之間的繼承關(guān)系。

圖片

圖片

再聊核心Sink

實(shí)際上Stream API內(nèi)部實(shí)現(xiàn)的的本質(zhì),就是如何重載Sink的這四個接口方法。

我還是從一個示例開始:

List<String> list = new ArrayList<>();
list.add("zhangsan");
list.add("ligeligeligeligeligeligeligeligeligelige");
list.add("lisilisilisilisilisilisilisilisi");
list.add("wangwu");
list.add("ligejishuligejishuligejishuligejishuligejishuligejishuligejishu");

List<String> resultList = list.stream()
.filter(it -> it.contains("li"))// 1. 只要包含li的數(shù)據(jù)
.filter(it -> it.contains("lige"))// 2. 只要包含lige的數(shù)據(jù)
.map(String::toUpperCase)// 3. 對符合的數(shù)據(jù)作進(jìn)一步加工,轉(zhuǎn)換大寫
.map(String::toLowerCase)// 4. 對符合的數(shù)據(jù)作進(jìn)一步加工,轉(zhuǎn)換小寫
.collect(Collectors.toList());

resultList.forEach(System.out::println);

不管是filter方法,還是map方法,還是其他的方法,我們進(jìn)入到源碼層面,返回了一個StatelessOp對象或StatefulOp對象。

所以便產(chǎn)生了這樣一個結(jié)構(gòu):

圖片

但是和Sink有什么關(guān)系呢?我們再反過來看filter或者map源碼:

圖片

直接返回一個匿名StatelessOp對象,實(shí)現(xiàn)opWrapSink方法,opWrapSink方法是傳入一個sink對象,返回另一個sink對象。而新的sink對象擁有傳入sink對象的引用。

但是,這個代碼有什么用?什么時候觸發(fā)的呢?

別著急,讓我們從collect(Collectors.toList())方法開始一步一步深入研究。

圖片

圖片

圖片

這里我們需要知道傳入xx方法的終端對象是ReduceOp,并且這個ReduceOp對象在makeSink的時候返回了一個匿名內(nèi)部類ReducingSink對象。

圖片

圖片

圖片

這里的makeSink我們提到過,返回一個匿名內(nèi)部類ReducingSink對象。

圖片

先執(zhí)行warpSink,再執(zhí)行copyInto。直白一點(diǎn)就是先對Sink進(jìn)行包裝成鏈?zhǔn)絊ink,再遍歷Sink鏈進(jìn)行copy到結(jié)果對象里。這里的兩個步驟都很核心。

先看warpSink:

圖片

首次進(jìn)入時,this為最后的Stream對象,從尾部向頭部遍歷

每次遍歷時,得到一個新的Stream對象,一般為StatelessOp對象或StatefulOp對象

執(zhí)行操作對象的opWrapSink方法,這就是匿名實(shí)現(xiàn)了。

在每一個opWrapSink實(shí)現(xiàn)方法中,傳入了上一個sink,最終得到一個sink鏈表

圖片

最后,返回Sink鏈的頭節(jié)點(diǎn),內(nèi)部稱之為包裝好的sink,命名wrapped,隨后,準(zhǔn)備進(jìn)行執(zhí)行begin,forEachRemaining,end方法。

圖片

forEachRemaning最終調(diào)用accept方法。

圖片

動畫理解Stream執(zhí)行流程

? 圖片 ?

責(zé)任編輯:武曉燕 來源: 李哥技術(shù)
相關(guān)推薦

2021-02-17 11:25:33

前端JavaScriptthis

2020-08-10 18:03:54

Cache存儲器CPU

2024-04-15 00:00:00

技術(shù)Attention架構(gòu)

2021-09-10 07:31:54

AndroidAppStartup原理

2022-11-04 09:43:05

Java線程

2024-03-12 00:00:00

Sora技術(shù)數(shù)據(jù)

2022-09-05 08:39:04

kubernetesk8s

2021-03-10 10:55:51

SpringJava代碼

2024-11-01 08:57:07

2021-09-08 06:51:52

AndroidRetrofit原理

2021-10-15 09:19:17

AndroidSharedPrefe分析源碼

2019-07-01 13:34:22

vue系統(tǒng)數(shù)據(jù)

2020-03-17 08:36:22

數(shù)據(jù)庫存儲Mysql

2023-10-13 13:30:00

MySQL鎖機(jī)制

2020-11-04 15:35:13

Golang內(nèi)存程序員

2023-09-19 22:47:39

Java內(nèi)存

2020-03-26 16:40:07

MySQL索引數(shù)據(jù)庫

2022-01-14 12:28:18

架構(gòu)OpenFeign遠(yuǎn)程

2022-09-26 08:01:31

線程LIFO操作方式

2009-11-16 17:20:04

PHP多維數(shù)組排序
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號