0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

Java 8 Stream流底層原理

jf_ro2CN3Fa ? 來源:CSDN ? 作者:CSDN ? 2022-11-18 10:27 ? 次閱讀

  • 函數(shù)式接口
    • 操作
    • 流程
    • Collection
    • AbstractPipeline
    • ReferencePipeline
    • Head
    • StatelessOp
    • StatefulOp
    • TerminalOp
    • ReduceOp
    • MatchOp
    • FindOp
    • ForEachOp
    • Sink
    • ChainedReference
    • TerminalSink
    • Collector
    • 并行流
    • ForkJoinTask
    • AbstractTask

函數(shù)式接口

初識lambda呢,函數(shù)式接口肯定是繞不過去的,函數(shù)式接口就是一個有且僅有一個抽象方法,但是可以有多個非抽象方法的接口。函數(shù)式接口可以被隱式轉(zhuǎn)換為lambda表達(dá)式。

@FunctionalInterface
publicinterfaceCloseable{

voidclose();
}

java.util.function它包含了很多類,用來支持Java的函數(shù)式編程,該包中的函數(shù)式接口有:

e1af6c02-66e6-11ed-8abf-dac502259ad0.png

操作

e1bb7024-66e6-11ed-8abf-dac502259ad0.png

流程

Stream相關(guān)接口繼承圖:

e1cd4740-66e6-11ed-8abf-dac502259ad0.png

Stream流水線組織結(jié)構(gòu)示意圖(圖是盜的):

e1dfbe7a-66e6-11ed-8abf-dac502259ad0.png

Collection

類路徑java.util.colltction

@Override
defaultSpliteratorspliterator(){
returnSpliterators.spliterator(this,0);
}
//常用Stream流轉(zhuǎn)換
defaultStreamstream(){
returnStreamSupport.stream(spliterator(),false);
}
//并行流
defaultStreamparallelStream(){
returnStreamSupport.stream(spliterator(),true);
}

//java.util.stream.StreamSupport#stream(java.util.Spliterator,boolean)
publicstaticStreamstream(Spliteratorspliterator,booleanparallel){
Objects.requireNonNull(spliterator);
returnnewReferencePipeline.Head<>(spliterator,StreamOpFlag.fromCharacteristics(spliterator),parallel);
}

AbstractPipeline

類路徑java.util.stream.AbstractPipeline

//反向鏈接到管道鏈的頭部(如果是源階段,則為自身)。
privatefinalAbstractPipelinesourceStage;

//“上游”管道,如果這是源階段,則為null。
privatefinalAbstractPipelinepreviousStage;

//此管道對象表示的中間操作的操作標(biāo)志。
protectedfinalintsourceOrOpFlags;

//管道中的下一個階段;如果這是最后一個階段,則為null。在鏈接到下一個管道時(shí)有效地結(jié)束。
privateAbstractPipelinenextStage;

//如果是順序的,則此管道對象與流源之間的中間操作數(shù);如果是并行的,則為先前有狀態(tài)的中間操作數(shù)。在管道準(zhǔn)備進(jìn)行評估時(shí)有效。
privateintdepth;

//源和所有操作的組合源標(biāo)志和操作標(biāo)志,直到此流水線對象表示的操作為止(包括該流水線對象所代表的操作)。在管道準(zhǔn)備進(jìn)行評估時(shí)有效。
privateintcombinedFlags;

//源拆分器。僅對頭管道有效。如果管道使用非null值,那么在使用管道之前,sourceSupplier必須為null。在使用管道之后,如果非null,則將其設(shè)置為null。
privateSpliteratorsourceSpliterator;

//來源供應(yīng)商。僅對頭管道有效。如果非null,則在使用管道之前,sourceSpliterator必須為null。在使用管道之后,如果非null,則將其設(shè)置為null。
privateSupplier>sourceSupplier;

//如果已鏈接或使用此管道,則為True
privatebooleanlinkedOrConsumed;

//如果正在執(zhí)行任何有狀態(tài)操作,則為true;否則為true。僅對源階段有效。
privatebooleansourceAnyStateful;

privateRunnablesourceCloseAction;

//如果管道是并行的,則為true;否則,管道為順序的;否則為true。僅對源階段有效。
privatebooleanparallel;

ReferencePipeline

類路徑:java.util.stream.ReferencePipeline

filter
//java.util.stream.ReferencePipeline#filter
@Override
publicfinalStreamfilter(PredicatesuperP_OUT>predicate){
Objects.requireNonNull(predicate);
//返回一個匿名無狀態(tài)的管道
returnnewStatelessOp(this,StreamShape.REFERENCE,StreamOpFlag.NOT_SIZED){
//下游生產(chǎn)線所需要的回調(diào)接口
@Override
SinkopWrapSink(intflags,Sinksink){
returnnewSink.ChainedReference(sink){
@Override
publicvoidbegin(longsize){
downstream.begin(-1);
}
//真正執(zhí)行操作的方法,依靠ChainedReference內(nèi)置ReferencePipeline引用下游的回調(diào)
@Override
publicvoidaccept(P_OUTu){
//只有滿足條件的元素才能被下游執(zhí)行
if(predicate.test(u))
downstream.accept(u);
}
};
}
};
}
map
//java.util.stream.ReferencePipeline#map
publicfinalStreammap(FunctionsuperP_OUT,?extendsR>mapper){
Objects.requireNonNull(mapper);
//返回一個匿名無狀態(tài)的管道
returnnewStatelessOp(this,StreamShape.REFERENCE,StreamOpFlag.NOT_SORTED|StreamOpFlag.NOT_DISTINCT){
//下游生產(chǎn)線所需要的回調(diào)接口
@Override
SinkopWrapSink(intflags,Sinksink){
returnnewSink.ChainedReference(sink){
//真正執(zhí)行操作的方法,依靠ChainedReference內(nèi)置ReferencePipeline引用下游的回調(diào)
@Override
publicvoidaccept(P_OUTu){
//執(zhí)行轉(zhuǎn)換后提供給下游執(zhí)行
downstream.accept(mapper.apply(u));
}
};
}
};
}
flatMap
//java.util.stream.ReferencePipeline#flatMap
@Override
publicfinalStreamflatMap(FunctionsuperP_OUT,?extendsStream>mapper){
Objects.requireNonNull(mapper);
//返回一個匿名無狀態(tài)的管道
returnnewStatelessOp(this,StreamShape.REFERENCE,
StreamOpFlag.NOT_SORTED|StreamOpFlag.NOT_DISTINCT|StreamOpFlag.NOT_SIZED){
//下游生產(chǎn)線所需要的回調(diào)接口
@Override
SinkopWrapSink(intflags,Sinksink){
returnnewSink.ChainedReference(sink){
@Override
publicvoidbegin(longsize){
downstream.begin(-1);
}
//真正執(zhí)行操作的方法,依靠ChainedReference內(nèi)置ReferencePipeline引用下游的回調(diào)
@Override
publicvoidaccept(P_OUTu){
try(Streamresult=mapper.apply(u)){
//劃分為多個流執(zhí)行下游(分流)
if(result!=null)
result.sequential().forEach(downstream);
}
}
};
}
};
}
peek
//java.util.stream.ReferencePipeline#peek
@Override
publicfinalStreampeek(ConsumersuperP_OUT>action){
Objects.requireNonNull(action);
//返回一個匿名無狀態(tài)的管道
returnnewStatelessOp(this,StreamShape.REFERENCE,0){
//下游生產(chǎn)線所需要的回調(diào)接口
@Override
SinkopWrapSink(intflags,Sinksink){
returnnewSink.ChainedReference(sink){
//真正執(zhí)行操作的方法,依靠ChainedReference內(nèi)置ReferencePipeline引用下游的回調(diào)
@Override
publicvoidaccept(P_OUTu){
//先執(zhí)行自定義方法,在執(zhí)行下游方法
action.accept(u);
downstream.accept(u);
}
};
}
};
}
sorted
@Override
publicfinalStreamsorted(){
//不提供Comparator,會使用元素自實(shí)現(xiàn)Comparator的compareTo方法
returnSortedOps.makeRef(this);
}

@Override
publicfinalStreamsorted(ComparatorsuperP_OUT>comparator){
returnSortedOps.makeRef(this,comparator);
}
//Sorted.makeRef
staticStreammakeRef(AbstractPipelineupstream,
ComparatorsuperT>comparator){
returnnewOfRef<>(upstream,comparator);
}
//ofRef類
privatestaticfinalclassOfRef<T>extendsReferencePipeline.StatefulOp<T,T>{

privatefinalbooleanisNaturalSort;
privatefinalComparatorsuperT>comparator;

@Override
publicSinkopWrapSink(intflags,Sinksink){
Objects.requireNonNull(sink);
//根據(jù)不同的flag進(jìn)行不同排序
if(StreamOpFlag.SORTED.isKnown(flags)&&isNaturalSort)
returnsink;
elseif(StreamOpFlag.SIZED.isKnown(flags))
returnnewSizedRefSortingSink<>(sink,comparator);
else
returnnewRefSortingSink<>(sink,comparator);
}

}
distinct
@Override
publicfinalStreamdistinct(){
returnDistinctOps.makeRef(this);
}
staticReferencePipelinemakeRef(AbstractPipelineupstream){
//返回一個匿名有狀態(tài)的管道
returnnewReferencePipeline.StatefulOp(upstream,StreamShape.REFERENCE,StreamOpFlag.IS_DISTINCT|StreamOpFlag.NOT_SIZED){

@Override
SinkopWrapSink(intflags,Sinksink){
Objects.requireNonNull(sink);

if(StreamOpFlag.DISTINCT.isKnown(flags)){
//已經(jīng)是去重過了
returnsink;
}elseif(StreamOpFlag.SORTED.isKnown(flags)){
//有序流
returnnewSink.ChainedReference(sink){
booleanseenNull;
//這個為先執(zhí)行的前序元素
TlastSeen;

@Override
publicvoidbegin(longsize){
seenNull=false;
lastSeen=null;
downstream.begin(-1);
}

@Override
publicvoidend(){
seenNull=false;
lastSeen=null;
downstream.end();
}
//這里通過有序的特性,前序元素與后序元素比較,如果相等則跳過執(zhí)行后序的元素
@Override
publicvoidaccept(Tt){
if(t==null){
//這里控制元素為null只有一個
if(!seenNull){
seenNull=true;
downstream.accept(lastSeen=null);
}
}elseif(lastSeen==null||!t.equals(lastSeen)){
//這里將前序元素賦值給lastSeen
downstream.accept(lastSeen=t);
}
}
};
}else{
//底層通過Set進(jìn)行去重,所以該元素需要重寫hashCode和equals方法
returnnewSink.ChainedReference(sink){
Setseen;

@Override
publicvoidbegin(longsize){
seen=newHashSet<>();
downstream.begin(-1);
}

@Override
publicvoidend(){
seen=null;
downstream.end();
}

@Override
publicvoidaccept(Tt){
if(!seen.contains(t)){
seen.add(t);
downstream.accept(t);
}
}
};
}
}
};
}
skip、limit
publicstaticStreammakeRef(AbstractPipelineupstream,
longskip,longlimit){
if(skip0)
thrownewIllegalArgumentException("Skipmustbenon-negative:"+skip);
//返回一個匿名有狀態(tài)的管道
returnnewReferencePipeline.StatefulOp(upstream,StreamShape.REFERENCE,flags(limit)){
SpliteratorunorderedSkipLimitSpliterator(Spliterators,longskip,longlimit,longsizeIfKnown){
if(skip<=?sizeIfKnown)?{
????????????????????limit?=?limit?>=0?Math.min(limit,sizeIfKnown-skip):sizeIfKnown-skip;
skip=0;
}
returnnewStreamSpliterators.UnorderedSliceSpliterator.OfRef<>(s,skip,limit);
}
//自己實(shí)現(xiàn)真正操作的方法
@Override
SinkopWrapSink(intflags,Sinksink){
returnnewSink.ChainedReference(sink){
longn=skip;
longm=limit>=0?limit:Long.MAX_VALUE;

@Override
publicvoidbegin(longsize){
downstream.begin(calcSize(size,skip,m));
}

@Override
publicvoidaccept(Tt){
if(n==0){
//limit
if(m>0){
m--;
downstream.accept(t);
}
}
//skip
else{
n--;
}
}

@Override
publicbooleancancellationRequested(){
returnm==0||downstream.cancellationRequested();
}
};
}
};
}
reduce
//java.util.stream.ReferencePipeline#reduce(P_OUT,java.util.function.BinaryOperator)
@Override
publicfinalP_OUTreduce(finalP_OUTidentity,finalBinaryOperatoraccumulator){
returnevaluate(ReduceOps.makeRef(identity,accumulator,accumulator));
}
//java.util.stream.ReferencePipeline#reduce(java.util.function.BinaryOperator)
@Override
publicfinalOptionalreduce(BinaryOperatoraccumulator){
returnevaluate(ReduceOps.makeRef(accumulator));
}
//java.util.stream.ReferencePipeline#reduce(R,java.util.function.BiFunction,java.util.function.BinaryOperator)
@Override
publicfinalRreduce(Ridentity,BiFunctionsuperP_OUT,R>accumulator,BinaryOperatorcombiner){
returnevaluate(ReduceOps.makeRef(identity,accumulator,combiner));
}

//java.util.stream.AbstractPipeline#evaluate(java.util.stream.TerminalOp)
finalRevaluate(TerminalOpterminalOp){
assertgetOutputShape()==terminalOp.inputShape();
if(linkedOrConsumed)
thrownewIllegalStateException(MSG_STREAM_LINKED);
linkedOrConsumed=true;

returnisParallel()
?terminalOp.evaluateParallel(this,sourceSpliterator(terminalOp.getOpFlags()))
:terminalOp.evaluateSequential(this,sourceSpliterator(terminalOp.getOpFlags()));
}
collect
//java.util.stream.ReferencePipeline#collect(java.util.stream.Collector)
@Override
@SuppressWarnings("unchecked")
publicfinalRcollect(CollectorsuperP_OUT,A,R>collector){
Acontainer;
if(isParallel()
&&(collector.characteristics().contains(Collector.Characteristics.CONCURRENT))
&&(!isOrdered()||collector.characteristics().contains(Collector.Characteristics.UNORDERED))){
container=collector.supplier().get();
BiConsumersuperP_OUT>accumulator=collector.accumulator();
forEach(u->accumulator.accept(container,u));
}
else{
container=evaluate(ReduceOps.makeRef(collector));
}
//具有特定轉(zhuǎn)換的使用finisher處理
returncollector.characteristics().contains(Collector.Characteristics.IDENTITY_FINISH)
?(R)container
:collector.finisher().apply(container);
}
//java.util.stream.ReferencePipeline#collect(java.util.function.Supplier,java.util.function.BiConsumer,java.util.function.BiConsumer)
@Override
publicfinalRcollect(Suppliersupplier,BiConsumersuperP_OUT>accumulator,BiConsumercombiner){
returnevaluate(ReduceOps.makeRef(supplier,accumulator,combiner));
}

//java.util.stream.AbstractPipeline#evaluate(java.util.stream.TerminalOp)
finalRevaluate(TerminalOpterminalOp){
assertgetOutputShape()==terminalOp.inputShape();
if(linkedOrConsumed)
thrownewIllegalStateException(MSG_STREAM_LINKED);
linkedOrConsumed=true;

returnisParallel()
?terminalOp.evaluateParallel(this,sourceSpliterator(terminalOp.getOpFlags()))
:terminalOp.evaluateSequential(this,sourceSpliterator(terminalOp.getOpFlags()));
}
forEach
//java.util.stream.ReferencePipeline#forEach
@Override
publicvoidforEach(ConsumersuperP_OUT>action){
evaluate(ForEachOps.makeRef(action,false));
}

//java.util.stream.ForEachOps#makeRef
publicstaticTerminalOpmakeRef(ConsumersuperT>action,booleanordered){
Objects.requireNonNull(action);
returnnewForEachOp.OfRef<>(action,ordered);
}

//java.util.stream.ForEachOps.ForEachOp.OfRef
staticfinalclassOfRef<T>extendsForEachOp<T>{
finalConsumersuperT>consumer;

OfRef(ConsumersuperT>consumer,booleanordered){
super(ordered);
this.consumer=consumer;
}

//只是簡單的消費(fèi)
@Override
publicvoidaccept(Tt){
consumer.accept(t);
}
}

Head

流的數(shù)據(jù)元的頭,類路徑java.util.stream.ReferencePipeline.Head

//java.util.stream.ReferencePipeline.Head
staticclassHead<E_IN,E_OUT>extendsReferencePipeline<E_IN,E_OUT>{

Head(Supplier>source,intsourceFlags,booleanparallel){
super(source,sourceFlags,parallel);
}

Head(Spliteratorsource,intsourceFlags,booleanparallel){
super(source,sourceFlags,parallel);
}

@Override
finalbooleanopIsStateful(){
thrownewUnsupportedOperationException();
}

@Override
finalSinkopWrapSink(intflags,Sinksink){
thrownewUnsupportedOperationException();
}
//Optimizedsequentialterminaloperationsfortheheadofthepipeline
@Override
publicvoidforEach(ConsumersuperE_OUT>action){
if(!isParallel()){
sourceStageSpliterator().forEachRemaining(action);
}
else{
super.forEach(action);
}
}

@Override
publicvoidforEachOrdered(ConsumersuperE_OUT>action){
if(!isParallel()){
sourceStageSpliterator().forEachRemaining(action);
}
else{
super.forEachOrdered(action);
}
}
}

StatelessOp

無狀態(tài)的中間管道,類路徑java.util.stream.ReferencePipeline.StatelessOp

//java.util.stream.ReferencePipeline.StatelessOp
abstractstaticclassStatelessOp<E_IN,E_OUT>extendsReferencePipeline<E_IN,E_OUT>{

StatelessOp(AbstractPipelineupstream,StreamShapeinputShape,intopFlags){
super(upstream,opFlags);
assertupstream.getOutputShape()==inputShape;
}

@Override
finalbooleanopIsStateful(){
returnfalse;
}
}

StatefulOp

有狀態(tài)的中間管道,類路徑java.util.stream.ReferencePipeline.StatefulOp

//java.util.stream.ReferencePipeline.StatefulOp
abstractstaticclassStatefulOp<E_IN,E_OUT>extendsReferencePipeline<E_IN,E_OUT>{

StatefulOp(AbstractPipelineupstream,StreamShapeinputShape,intopFlags){
super(upstream,opFlags);
assertupstream.getOutputShape()==inputShape;
}

@Override
finalbooleanopIsStateful(){
returntrue;
}

@Override
abstractNodeopEvaluateParallel(PipelineHelperhelper,
Spliteratorspliterator,
IntFunctiongenerator);

TerminalOp

管道流的結(jié)束操作,類路徑java.util.stream.TerminalOp

interfaceTerminalOp<E_IN,R>{

//獲取此操作的輸入類型的形狀
defaultStreamShapeinputShape(){returnStreamShape.REFERENCE;}

//獲取操作的流標(biāo)志。終端操作可以設(shè)置StreamOpFlag定義的流標(biāo)志的有限子集,并且這些標(biāo)志與管道的先前組合的流和中間操作標(biāo)志組合在一起。
defaultintgetOpFlags(){return0;}

//使用指定的PipelineHelper對操作執(zhí)行并行評估,該操作描述上游中間操作。
defaultRevaluateParallel(PipelineHelperhelper,Spliteratorspliterator){
if(Tripwire.ENABLED)
Tripwire.trip(getClass(),"{0}triggeringTerminalOp.evaluateParallelserialdefault");
returnevaluateSequential(helper,spliterator);
}

//使用指定的PipelineHelper對操作執(zhí)行順序評估,該操作描述上游中間操作。
RevaluateSequential(PipelineHelperhelper,Spliteratorspliterator);
}

ReduceOp

類路徑java.util.stream.ReduceOps.ReduceOp

privatestaticabstractclassReduceOp<T,R,SextendsAccumulatingSink<T,R,S>>implementsTerminalOp<T,R>{
privatefinalStreamShapeinputShape;

ReduceOp(StreamShapeshape){
inputShape=shape;
}

publicabstractSmakeSink();

@Override
publicStreamShapeinputShape(){
returninputShape;
}

//通過匿名子類實(shí)現(xiàn)makeSink()獲取Sink
@Override
publicRevaluateSequential(PipelineHelperhelper,Spliteratorspliterator){
returnhelper.wrapAndCopyInto(makeSink(),spliterator).get();
}

@Override
publicRevaluateParallel(PipelineHelperhelper,Spliteratorspliterator){
returnnewReduceTask<>(this,helper,spliterator).invoke().get();
}
}

MatchOp

類路徑java.util.stream.MatchOps.MatchOp

privatestaticfinalclassMatchOp<T>implementsTerminalOp<T,Boolean>{
privatefinalStreamShapeinputShape;
finalMatchKindmatchKind;
finalSupplier>sinkSupplier;

MatchOp(StreamShapeshape,MatchKindmatchKind,Supplier>sinkSupplier){
this.inputShape=shape;
this.matchKind=matchKind;
this.sinkSupplier=sinkSupplier;
}

@Override
publicintgetOpFlags(){
returnStreamOpFlag.IS_SHORT_CIRCUIT|StreamOpFlag.NOT_ORDERED;
}

@Override
publicStreamShapeinputShape(){
returninputShape;
}

//使用內(nèi)置的sinkSupplier獲取Sink
@Override
publicBooleanevaluateSequential(PipelineHelperhelper,Spliteratorspliterator){
returnhelper.wrapAndCopyInto(sinkSupplier.get(),spliterator).getAndClearState();
}

@Override
publicBooleanevaluateParallel(PipelineHelperhelper,Spliteratorspliterator){
returnnewMatchTask<>(this,helper,spliterator).invoke();
}
}

FindOp

類路徑java.util.stream.FindOps.FindOp

privatestaticfinalclassFindOp<T,O>implementsTerminalOp<T,O>{
privatefinalStreamShapeshape;
finalbooleanmustFindFirst;
finalOemptyValue;
finalPredicatepresentPredicate;
finalSupplier>sinkSupplier;

FindOp(booleanmustFindFirst,
StreamShapeshape,
OemptyValue,
PredicatepresentPredicate,
Supplier>sinkSupplier){
this.mustFindFirst=mustFindFirst;
this.shape=shape;
this.emptyValue=emptyValue;
this.presentPredicate=presentPredicate;
this.sinkSupplier=sinkSupplier;
}

@Override
publicintgetOpFlags(){
returnStreamOpFlag.IS_SHORT_CIRCUIT|(mustFindFirst?0:StreamOpFlag.NOT_ORDERED);
}

@Override
publicStreamShapeinputShape(){
returnshape;
}

//通過內(nèi)置sinkSupplier獲取Sink
@Override
publicOevaluateSequential(PipelineHelperhelper,Spliteratorspliterator){
Oresult=helper.wrapAndCopyInto(sinkSupplier.get(),spliterator).get();
returnresult!=null?result:emptyValue;
}

@Override
publicOevaluateParallel(PipelineHelperhelper,Spliteratorspliterator){
returnnewFindTask<>(this,helper,spliterator).invoke();
}
}

ForEachOp

類路徑java.util.stream.ForEachOps.ForEachOp

staticabstractclassForEachOp<T>implementsTerminalOp<T,Void>,TerminalSink<T,Void>{
privatefinalbooleanordered;

protectedForEachOp(booleanordered){
this.ordered=ordered;
}

@Override
publicintgetOpFlags(){
returnordered?0:StreamOpFlag.NOT_ORDERED;
}

//自己實(shí)現(xiàn)了Sink
@Override
publicVoidevaluateSequential(PipelineHelperhelper,Spliteratorspliterator){
returnhelper.wrapAndCopyInto(this,spliterator).get();
}

@Override
publicVoidevaluateParallel(PipelineHelperhelper,Spliteratorspliterator){
if(ordered)
newForEachOrderedTask<>(helper,spliterator,this).invoke();
else
newForEachTask<>(helper,spliterator,helper.wrapSink(this)).invoke();
returnnull;
}

@Override
publicVoidget(){
returnnull;
}

staticfinalclassOfRef<T>extendsForEachOp<T>{
finalConsumersuperT>consumer;

OfRef(ConsumersuperT>consumer,booleanordered){
super(ordered);
this.consumer=consumer;
}

@Override
publicvoidaccept(Tt){
consumer.accept(t);
}
}
...
}

Sink

類路徑java.util.stream.Sink

interfaceSink<T>extendsConsumer<T>{
//開始遍歷元素之前調(diào)用該方法,通知Sink做好準(zhǔn)備。
defaultvoidbegin(longsize){}
//所有元素遍歷完成之后調(diào)用,通知Sink沒有更多的元素了。
defaultvoidend(){}
//是否可以結(jié)束操作,可以讓短路操作盡早結(jié)束。
defaultbooleancancellationRequested(){
returnfalse;
}
//遍歷元素時(shí)調(diào)用,接受一個待處理元素,并對元素進(jìn)行處理。Stage把自己包含的操作和回調(diào)方法封裝到該方法里,前一個Stage只需要調(diào)用當(dāng)前Stage.accept(Tt)方法就行了。
voidaccept(Tt);
}

這里Sink的子類實(shí)現(xiàn)中分為兩種:中間操作匿名實(shí)現(xiàn)ChainedReferenceTerminalOp子類所提供的Sink。

ChainedReference

類路徑java.util.stream.Sink.ChainedReference,這里是中間操作的默認(rèn)模板父類

staticabstractclassChainedReference<T,E_OUT>implementsSink<T>{
protectedfinalSinksuperE_OUT>downstream;

publicChainedReference(SinksuperE_OUT>downstream){
this.downstream=Objects.requireNonNull(downstream);
}

@Override
publicvoidbegin(longsize){
downstream.begin(size);
}

@Override
publicvoidend(){
downstream.end();
}

@Override
publicbooleancancellationRequested(){
returndownstream.cancellationRequested();
}
}

在上述的中間操作管道流中都是通過匿名類繼承ChainedReference實(shí)現(xiàn)onWrapSink(int, Sink)返回一個指定操作的Sink。

TerminalSink

這里為什么講提供呢?這是因?yàn)椴煌膶?shí)現(xiàn)TerminalOp的子類中在實(shí)現(xiàn)java.util.stream.TerminalOp#evaluateSequential中都是通過helper.wrapAndCopyInto(TerminalOp子類實(shí)現(xiàn)提供的Sink, spliterator)中通過參數(shù)傳遞的方式提供的,不同的子類傳遞的方式不一樣所以此處用了一個提供Sink

由ReduceOps中實(shí)現(xiàn)TerminalOp所提供的ReducingSink,它是由匿名類實(shí)現(xiàn)java.util.stream.ReduceOps.ReduceOp#makeSink來交付給helper.wrapAndCopyInto(makeSink(), spliterator)的。

publicstaticTerminalOpmakeRef(Useed,BiFunctionsuperT,U>reducer,BinaryOperatorcombiner){
Objects.requireNonNull(reducer);
Objects.requireNonNull(combiner);
classReducingSinkextendsBox<U>implementsAccumulatingSink<T,U,ReducingSink>{
@Override
publicvoidbegin(longsize){
state=seed;
}

@Override
publicvoidaccept(Tt){
state=reducer.apply(state,t);
}

@Override
publicvoidcombine(ReducingSinkother){
state=combiner.apply(state,other.state);
}
}
returnnewReduceOp(StreamShape.REFERENCE){
@Override
publicReducingSinkmakeSink(){
returnnewReducingSink();
}
};
}

ForEachOps中實(shí)現(xiàn)TerminalOp所提供的是this,它的提供方式就是通過this交付給helper.wrapAndCopyInto(this, spliterator)。

//這里ForEachOp自己通過TerminalSink間接的實(shí)現(xiàn)了Sink
staticabstractclassForEachOp<T>implementsTerminalOp<T,Void>,TerminalSink<T,Void>{
@Override
publicVoidevaluateSequential(PipelineHelperhelper,Spliteratorspliterator){
returnhelper.wrapAndCopyInto(this,spliterator).get();
}
}

由MatchOps中實(shí)現(xiàn)TerminalOp所提供的sinkSupplier通過構(gòu)造函數(shù)由外部賦值,通過Supplier接口的get()來交付給helper.wrapAndCopyInto(sinkSupplier.get(), spliterator)

privatestaticfinalclassMatchOp<T>implementsTerminalOp<T,Boolean>{
finalSupplier>sinkSupplier;

@Override
publicBooleanevaluateSequential(PipelineHelperhelper,Spliteratorspliterator){
returnhelper.wrapAndCopyInto(sinkSupplier.get(),spliterator).getAndClearState();
}
}

由FindOps中實(shí)現(xiàn)TerminalOp所提供的與上述MatchOps是一致的

privatestaticfinalclassFindOp<T,O>implementsTerminalOp<T,O>{
finalSupplier>sinkSupplier;

@Override
publicOevaluateSequential(PipelineHelperhelper,Spliteratorspliterator){
Oresult=helper.wrapAndCopyInto(sinkSupplier.get(),spliterator).get();
returnresult!=null?result:emptyValue;
}
}

Collector

在Collector中有以下幾個實(shí)現(xiàn)接口:

  • Supplier:結(jié)果類型的提供器。
  • BiConsumer:將元素放入結(jié)果的累加器。
  • BinaryOperator:合并部分結(jié)果的組合器。
  • Function:對結(jié)果類型轉(zhuǎn)換為最終結(jié)果類型的轉(zhuǎn)換器。
  • Set:保存Collector特征的集合

并行流

前述都是基于串行流的講解,其實(shí)并行流也是基于上述的helper.wrapAndCopyInto(op.sinkSupplier.get(), spliterator)這個方法上面做的一層基于ForkJoinTask多線程框架的封裝。

ForkJoinTask

ForkJoin框架的思想就是分而治之,它將一個大任務(wù)切割為多個小任務(wù)這個過程稱為fork,將每個任務(wù)的執(zhí)行的結(jié)果進(jìn)行匯總的過程稱為join。ForkJoin框架相關(guān)的接口關(guān)系圖如下(圖是盜的):

e1fb7a7a-66e6-11ed-8abf-dac502259ad0.png

AbstractTask

類路徑java.util.stream.AbstractTask,AbstractTask繼承了在JUC中已經(jīng)封裝好的ForkJoinTask抽象子類java.util.concurrent.CountedCompleter

此類基于CountedCompleter ,它是fork-join任務(wù)的一種形式,其中每個任務(wù)都有未完成子代的信號量計(jì)數(shù),并且該任務(wù)隱式完成并在其最后一個子代完成時(shí)得到通知。 內(nèi)部節(jié)點(diǎn)任務(wù)可能會覆蓋CountedCompleteronCompletion方法,以將子任務(wù)的結(jié)果合并到當(dāng)前任務(wù)的結(jié)果中。

拆分和設(shè)置子任務(wù)鏈接是由內(nèi)部節(jié)點(diǎn)的compute()完成的。 在葉節(jié)點(diǎn)的compute()時(shí)間,可以確保將為所有子代設(shè)置父代的子代相關(guān)字段(包括父代子代的同級鏈接)。

例如,執(zhí)行減少任務(wù)的任務(wù)將覆蓋doLeaf()以使用Spliterator對該葉節(jié)點(diǎn)的塊執(zhí)行減少Spliterator ,并覆蓋onCompletion()以合并內(nèi)部節(jié)點(diǎn)的子任務(wù)的結(jié)果:

@Override
protectedReduceTaskmakeChild(Spliteratorspliterator){
//返回一個ForkJoinTask任務(wù)
returnnewReduceTask<>(this,spliterator);
}

@Override
protectedSdoLeaf(){
//其他實(shí)現(xiàn)大同小異
returnhelper.wrapAndCopyInto(op.makeSink(),spliterator);
}

@Override
publicvoidonCompletion(CountedCompletercaller){
//非葉子節(jié)點(diǎn)進(jìn)行結(jié)果組合
if(!isLeaf()){
SleftResult=leftChild.getLocalResult();
leftResult.combine(rightChild.getLocalResult());
setLocalResult(leftResult);
}
//GCspliterator,leftandrightchild
super.onCompletion(caller);
}

AbstractTask封裝了分片任務(wù)的算法模板,通過是SpliteratortrySplit()方法來實(shí)現(xiàn)分片的細(xì)節(jié),詳細(xì)算法源碼如下(類路徑:java.util.stream.AbstractTask#compute):

@Override
publicvoidcompute(){
//將當(dāng)前這個spliterator作為右節(jié)點(diǎn)(此時(shí)為root節(jié)點(diǎn))
Spliteratorrs=spliterator,ls;
//評估任務(wù)的大小
longsizeEstimate=rs.estimateSize();
//獲取任務(wù)閾值
longsizeThreshold=getTargetSize(sizeEstimate);
booleanforkRight=false;
@SuppressWarnings("unchecked")Ktask=(K)this;
//細(xì)節(jié)不多贅述,下面我用圖來講解算法
/**
*根節(jié)點(diǎn)指定為:右邊節(jié)點(diǎn)
*root
*split()
*leftright
*left.fork()
*split()
*lr
*rs=ls
*right.fork()
*split()
*lr
*l.fork()
*/
while(sizeEstimate>sizeThreshold&&(ls=rs.trySplit())!=null){
KleftChild,rightChild,taskToFork;
task.leftChild=leftChild=task.makeChild(ls);
task.rightChild=rightChild=task.makeChild(rs);
task.setPendingCount(1);
if(forkRight){
forkRight=false;
//左右節(jié)點(diǎn)切換進(jìn)行fork和split
rs=ls;
task=leftChild;
taskToFork=rightChild;
}
else{
forkRight=true;
task=rightChild;
taskToFork=leftChild;
}
//fork任務(wù)加入隊(duì)列中去
taskToFork.fork();
sizeEstimate=rs.estimateSize();
}
//將執(zhí)行doLeaf底層就是單個串行流的操作
task.setLocalResult(task.doLeaf());
//將結(jié)果組合成一個最終結(jié)果
task.tryComplete();
}

AbstractTask執(zhí)行與分片流程圖如下:

e20b4e46-66e6-11ed-8abf-dac502259ad0.png

到這里Stream流的相關(guān)知識介紹到這,這里附上一副總體圖來加深下印象

e217cc8e-66e6-11ed-8abf-dac502259ad0.png

審核編輯 :李倩


聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報(bào)投訴
  • 接口
    +關(guān)注

    關(guān)注

    33

    文章

    8615

    瀏覽量

    151306
  • JAVA
    +關(guān)注

    關(guān)注

    19

    文章

    2970

    瀏覽量

    104814
  • 函數(shù)
    +關(guān)注

    關(guān)注

    3

    文章

    4333

    瀏覽量

    62700

原文標(biāo)題:還有人不知道 Java 8 Stream流底層原理?

文章出處:【微信號:芋道源碼,微信公眾號:芋道源碼】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏

    評論

    相關(guān)推薦

    Java 23功能介紹

    Java 23 包含全新和更新的 Java 語言功能、核心 API 以及 JVM,同時(shí)適合新的 Java 開發(fā)者和高級開發(fā)者。從?IntelliJ IDEA 2024.2?開始已支持 Java
    的頭像 發(fā)表于 12-04 10:02 ?235次閱讀
    <b class='flag-5'>Java</b> 23功能介紹

    Java集合API的改進(jìn)介紹

    解答這些問題。 我們將逐步學(xué)習(xí) Java 集合類的優(yōu)化過程,并按版本逐一對比分析。主要討論的焦點(diǎn)將包括 JDK 1.0、1.2、1.4、1.5、1.6、1.8、9、10、11 和 21 版本的 Java 集合功能 Java 集合
    的頭像 發(fā)表于 11-22 11:12 ?213次閱讀
    <b class='flag-5'>Java</b>集合API的改進(jìn)介紹

    Java中時(shí)間戳的使用

    Java中時(shí)間戳的使用
    的頭像 發(fā)表于 11-06 16:04 ?226次閱讀
    <b class='flag-5'>Java</b>中時(shí)間戳的使用

    java反編譯能拿到源碼嗎

    Java反編譯是一種將編譯后的Java字節(jié)碼(.class文件)轉(zhuǎn)換回Java源代碼的過程。雖然反編譯可以幫助理解代碼的邏輯和結(jié)構(gòu),但它并不總是能完美地還原原始源代碼。反編譯工具通常會產(chǎn)生與原始代碼
    的頭像 發(fā)表于 09-02 11:03 ?1043次閱讀

    如何成功的烘烤微控SU-8光刻膠?

    在微控PDMS芯片加工的過程中,需要使用烘膠臺或者烤膠設(shè)備對SU-8光刻膠或PDMS聚合物進(jìn)行烘烤。SU-8光刻膠的烘烤通常需要進(jìn)行2-3次。本文簡要介紹SU-8光刻膠烘烤的注意事項(xiàng)
    的頭像 發(fā)表于 08-27 15:54 ?302次閱讀

    華納云:java web和java有什么區(qū)別java web和java有什么區(qū)別

    Java Web和Java是兩個不同的概念,它們在功能、用途和實(shí)現(xiàn)方式上存在一些區(qū)別,下面將詳細(xì)介紹它們之間的區(qū)別。 1. 功能和用途: – Java是一種編程語言,它提供了一種用于開發(fā)各種應(yīng)用程序
    的頭像 發(fā)表于 07-16 13:35 ?825次閱讀
    華納云:<b class='flag-5'>java</b> web和<b class='flag-5'>java</b>有什么區(qū)別<b class='flag-5'>java</b> web和<b class='flag-5'>java</b>有什么區(qū)別

    ESP-ADF下的i2s_stream是否可以全雙工工作?

    請問各位朋友: ESP-ADF下的i2s_stream是否可以全雙工工作,我看了其下的所有關(guān)于I2S的例子程序和create_i2s_stream函數(shù)的源碼,都只能單工工作(要么讀要么寫),我要自行修改與i2s_stream相關(guān)
    發(fā)表于 06-28 06:59

    如何在STM32F439 DMA中斷中區(qū)分是哪個通道產(chǎn)生的中斷?

    STM32F439有兩個DMA控制器DMA1和DMA2,每個控制器有8,每個8個通道,我做的是采用串口1DMA中斷接收數(shù)據(jù),DMA2的
    發(fā)表于 05-17 08:20

    Oracle確認(rèn)Java/JDK 11官方支持延長至2032年1月?

    此外,Solaris操作系統(tǒng)上的Java SE 8Java SE 11的官方支持也同步延期至2030年12月及2032年1月,進(jìn)一步延長了該平臺上的Java服務(wù)周期。
    的頭像 發(fā)表于 05-16 15:57 ?1280次閱讀

    已經(jīng)安裝了Java,且依然提示安裝Java是為什么?

    我已經(jīng)在機(jī)器上安裝了最新版的 Java 10,打開 Cube 卻得到要求安裝 Java 1.7.0_45 的提示。何解?Eclipse CDT 依賴 Java,不可卸載重裝。
    發(fā)表于 04-26 06:23

    使用STM32F429的DMA多個外設(shè)都使用到同樣的DMA_STREAM的時(shí)候,就會發(fā)生沖突怎么解決?

    最近在使用STM32F429的DMA時(shí)候,發(fā)現(xiàn)一個問題,當(dāng)多個外設(shè)都使用到同樣的DMA_STREAM的時(shí)候,就會發(fā)生沖突(后面配置的DMA可用,前面配置的不能用),我用的USART6_TX用
    發(fā)表于 04-24 07:13

    stm32F429串口采用DMA方式發(fā)送,數(shù)據(jù)使能失敗的原因?

    DMA1 時(shí)鐘穩(wěn)定 DMA_DeInit(DMA2_Stream7);// 復(fù)位初始化DMA數(shù)據(jù) while (DMA_GetCmdStatus(DMA2_Stream7) != DISABLE
    發(fā)表于 04-17 07:05

    java實(shí)現(xiàn)多線程的幾種方式

    了多種實(shí)現(xiàn)多線程的方式,本文將詳細(xì)介紹以下幾種方式: 1.繼承Thread類 2.實(shí)現(xiàn)Runnable接口 3.Callable和Future 4.線程池 5.Java 8
    的頭像 發(fā)表于 03-14 16:55 ?723次閱讀

    Oracle 2024年Java發(fā)展藍(lán)圖分析

    Oracle 的 Java 開發(fā)者布道師 Nicolai Parlog 于近日發(fā)布一段視頻,介紹了 2024 年的 Java 工作規(guī)劃。
    的頭像 發(fā)表于 01-26 14:27 ?1349次閱讀

    XMC Pinout Tool是否可以不受限制地使用免費(fèi)的Java版本運(yùn)行?

    ,我想問一下 Java 許可證是否適用于例如 Java Update 8 381 包含在 Pinout Tool 中,因?yàn)榇?Java 版本通常會產(chǎn)生許可費(fèi)用。
    發(fā)表于 01-22 06:18