Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
functionalj.stream.DoubleStreamPlus Maven / Gradle / Ivy
// ============================================================================
// Copyright (c) 2017-2019 Nawapunth Manusitthipol (NawaMan - http://nawaman.net).
// ----------------------------------------------------------------------------
// MIT License
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in all
// copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
// SOFTWARE.
// ============================================================================
package functionalj.stream;
import java.util.ArrayList;
import java.util.DoubleSummaryStatistics;
import java.util.List;
import java.util.OptionalDouble;
import java.util.PrimitiveIterator;
import java.util.Set;
import java.util.Spliterator;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.atomic.AtomicReference;
import java.util.function.BiConsumer;
import java.util.function.DoubleBinaryOperator;
import java.util.function.DoubleConsumer;
import java.util.function.DoubleFunction;
import java.util.function.DoublePredicate;
import java.util.function.DoubleSupplier;
import java.util.function.DoubleToIntFunction;
import java.util.function.DoubleToLongFunction;
import java.util.function.DoubleUnaryOperator;
import java.util.function.Function;
import java.util.function.ObjDoubleConsumer;
import java.util.function.Supplier;
import java.util.stream.Collectors;
import java.util.stream.DoubleStream;
import java.util.stream.Stream;
import functionalj.function.Func1;
import functionalj.function.FuncUnit1;
import functionalj.functions.StrFuncs;
import functionalj.list.FuncList;
import functionalj.list.ImmutableList;
import functionalj.pipeable.Pipeable;
import functionalj.tuple.Tuple;
import functionalj.tuple.Tuple2;
import functionalj.tuple.Tuple3;
import functionalj.tuple.Tuple4;
import functionalj.tuple.Tuple5;
import functionalj.tuple.Tuple6;
import lombok.val;
@FunctionalInterface
public interface DoubleStreamPlus extends DoubleStream {
public static DoubleStreamPlus of(double ... doubles) {
return DoubleStreamPlus.from(DoubleStream.of(doubles));
}
public static DoubleStreamPlus from(DoubleStream doubleStream) {
return ()->doubleStream;
}
public static DoubleStreamPlus concat(DoubleStreamPlus ... streams) {
return DoubleStreamPlus.from(StreamPlus.of(streams).flatMap(s -> s.asStream()).mapToDouble(i -> i));
}
public static DoubleStreamPlus empty() {
return DoubleStreamPlus.from(DoubleStream.empty());
}
public static DoubleStreamPlus infinite() {
return DoubleStreamPlus.from(DoubleStream.iterate(0, i -> i + 1));
}
public static DoubleStreamPlus generate(DoubleSupplier s) {
return DoubleStreamPlus.from(DoubleStream.generate(s));
}
public static DoubleStreamPlus iterate(double seed, DoubleUnaryOperator f) {
return DoubleStreamPlus.from(DoubleStream.iterate(seed, f));
}
public static DoubleStreamPlus compound(double seed, DoubleUnaryOperator f) {
return DoubleStreamPlus.from(DoubleStream.iterate(seed, f));
}
public static DoubleStreamPlus iterate(double seed1, double seed2, DoubleBinaryOperator f) {
val counter = new AtomicInteger(0);
val int1 = new AtomicReference(seed1);
val int2 = new AtomicReference(seed2);
return DoubleStreamPlus.generate(()->{
if (counter.getAndIncrement() == 0)
return seed1;
if (counter.getAndIncrement() == 2)
return seed2;
double i2 = int2.get();
double i1 = int1.getAndSet(i2);
double i = f.applyAsDouble(i1, i2);
int2.set(i);
return i;
});
}
//== Stream ==
public DoubleStream stream();
public default TARGET terminate(Func1 action) {
val stream = stream();
try {
val result = action.apply(stream);
return result;
} finally {
stream.close();
}
}
public default void terminate(FuncUnit1 action) {
val stream = stream();
try {
action.accept(stream);
} finally {
stream.close();
}
}
@Override
public default DoubleStreamPlus filter(DoublePredicate predicate) {
return DoubleStreamPlus.from(stream().filter(predicate));
}
@Override
public default DoubleStreamPlus map(DoubleUnaryOperator mapper) {
return DoubleStreamPlus.from(stream().map(mapper));
}
public default Pipeable pipable() {
return Pipeable.of(this);
}
public default T pipe(Function super DoubleStreamPlus, T> piper) {
return piper.apply(this);
}
public default StreamPlus mapBy(DoubleFunction extends U> mapper) {
return StreamPlus.from(stream().mapToObj(mapper));
}
public default StreamPlus asStream() {
val stream = StreamPlus.from(stream()
.mapToObj(i -> Double.valueOf(i)));
stream.onClose(()->{ close(); });
return stream;
}
@Override
public default StreamPlus mapToObj(DoubleFunction extends U> mapper) {
StreamPlus stream = StreamPlus.from(stream().mapToObj(mapper));
stream.onClose(()->{ close(); });
return stream;
}
public default StreamPlus mapToObj(Supplier extends TARGET> supplier) {
StreamPlus stream = StreamPlus.from(stream().mapToObj(e -> supplier.get()));
stream.onClose(()->{ close(); });
return stream;
}
@Override
public default IntStreamPlus mapToInt(DoubleToIntFunction mapper) {
return IntStreamPlus.from(stream().mapToInt(mapper));
}
@Override
public default LongStreamPlus mapToLong(DoubleToLongFunction mapper) {
return LongStreamPlus.from(stream().mapToLong(mapper));
}
@Override
public default DoubleStreamPlus flatMap(DoubleFunction extends DoubleStream> mapper) {
return DoubleStreamPlus.from(stream().flatMap(mapper));
}
public default DoubleStreamPlus flatMap(Function> mapper) {
return DoubleStreamPlus.from(stream()
.mapToObj (i -> Double.valueOf(i))
.flatMap (i -> mapper.apply(i))
.mapToDouble(i -> i.doubleValue()));
}
@Override
public default DoubleStreamPlus distinct() {
return DoubleStreamPlus.from(stream().distinct());
}
@Override
public default DoubleStreamPlus sorted() {
return DoubleStreamPlus.from(stream().sorted());
}
@Override
public default DoubleStreamPlus peek(DoubleConsumer action) {
return DoubleStreamPlus.from(stream().peek(action));
}
@Override
public default DoubleStreamPlus limit(long maxSize) {
return DoubleStreamPlus.from(stream().limit(maxSize));
}
@Override
public default DoubleStreamPlus skip(long n) {
return DoubleStreamPlus.from(stream().skip(n));
}
@Override
public default void forEach(DoubleConsumer action) {
terminate(stream-> {
stream
.forEach(action);
});
}
@Override
public default void forEachOrdered(DoubleConsumer action) {
terminate(stream-> {
stream
.forEachOrdered(action);
});
}
@Override
public default double[] toArray() {
return terminate(stream-> {
return stream
.toArray();
});
}
@Override
public default double reduce(double identity, DoubleBinaryOperator op) {
return terminate(stream-> {
return stream
.reduce(identity, op);
});
}
@Override
public default OptionalDouble reduce(DoubleBinaryOperator op) {
return terminate(stream-> {
return stream
.reduce(op);
});
}
@Override
public default R collect(Supplier supplier,
ObjDoubleConsumer accumulator,
BiConsumer combiner) {
return terminate(stream-> {
return stream
.collect(supplier, accumulator, combiner);
});
}
@Override
public default double sum() {
return terminate(stream-> {
return stream
.sum();
});
}
@Override
public default OptionalDouble min() {
return terminate(stream-> {
return stream
.min();
});
}
@Override
public default OptionalDouble max() {
return terminate(stream-> {
return stream
.max();
});
}
@Override
public default long count() {
return terminate(stream-> {
return stream
.count();
});
}
@Override
public default OptionalDouble average() {
return terminate(stream-> {
return stream
.average();
});
}
@Override
public default DoubleSummaryStatistics summaryStatistics() {
return terminate(stream-> {
return stream
.summaryStatistics();
});
}
@Override
public default boolean anyMatch(DoublePredicate predicate) {
return terminate(stream-> {
return stream
.anyMatch(predicate);
});
}
@Override
public default boolean allMatch(DoublePredicate predicate) {
return terminate(stream-> {
return stream
.allMatch(predicate);
});
}
@Override
public default boolean noneMatch(DoublePredicate predicate) {
return terminate(stream-> {
return stream
.noneMatch(predicate);
});
}
@Override
public default OptionalDouble findFirst() {
return terminate(stream-> {
return stream
.findFirst();
});
}
@Override
public default OptionalDouble findAny() {
return terminate(stream-> {
return stream
.findAny();
});
}
@Override
public default StreamPlus boxed() {
return StreamPlus.from(stream().boxed());
}
@Override
public default DoubleStreamPlus sequential() {
return DoubleStreamPlus.from(stream().sequential());
}
@Override
public default DoubleStreamPlus parallel() {
return DoubleStreamPlus.from(stream().sequential());
}
@Override
public default PrimitiveIterator.OfDouble iterator() {
// TODO - Make sure close is handled properly.
return stream().iterator();
}
@Override
public default Spliterator.OfDouble spliterator() {
return stream().spliterator();
}
@Override
public default boolean isParallel() {
return stream().isParallel();
}
@Override
public default DoubleStreamPlus unordered() {
return DoubleStreamPlus.from(stream().unordered());
}
@Override
public default DoubleStreamPlus onClose(Runnable closeHandler) {
return DoubleStreamPlus.from(stream().onClose(closeHandler));
}
@Override
public default void close() {
stream().close();
}
//== Additional functionalities
public default StreamPlus segment(int count) {
return segment(count, true);
}
public default StreamPlus segment(int count, boolean includeTail) {
// if (count <= 1)
// return this;
val index = new AtomicInteger(0);
return segment(data -> (index.getAndIncrement() % count) == 0, includeTail);
}
public default StreamPlus segment(DoublePredicate startCondition) {
return segment(startCondition, true);
}
public default StreamPlus segment(DoublePredicate startCondition, boolean includeTail) {
val list = new AtomicReference<>(new ArrayList());
val adding = new AtomicBoolean(false);
StreamPlus mainStream = StreamPlus.from(
mapToObj(i ->{
if (startCondition.test(i)) {
adding.set(true);
val retList = list.getAndUpdate(l -> new ArrayList());
list.get().add(i);
if (retList.isEmpty())
return null;
return DoubleStreamPlus.from(retList.stream().mapToDouble(Double::doubleValue));
}
if (adding.get()) list.get().add(i);
return null;
}))
.filterNonNull();
;
val mainSupplier = (Supplier>)()->mainStream;
if (!includeTail)
return mainStream;
val tailSupplier = (Supplier>)()->{
return StreamPlus.of(
DoubleStreamPlus.from(
list.get()
.stream()
.mapToDouble(Double::doubleValue)));
};
val resultStream = StreamPlus.of(
mainSupplier,
tailSupplier
)
.flatMap(Supplier::get);
return resultStream;
}
public default StreamPlus segment(DoublePredicate startCondition, DoublePredicate endCondition) {
return segment(startCondition, endCondition, true);
}
public default StreamPlus segment(DoublePredicate startCondition, DoublePredicate endCondition, boolean includeTail) {
val list = new AtomicReference<>(new ArrayList());
val adding = new AtomicBoolean(false);
StreamPlus stream = StreamPlus.from(
mapToObj(i ->{
if (startCondition.test(i)) {
adding.set(true);
}
if (includeTail && adding.get()) list.get().add(i);
if (endCondition.test(i)) {
adding.set(false);
val retList = list.getAndUpdate(l -> new ArrayList());
return DoubleStreamPlus.from(retList.stream().mapToDouble(Double::doubleValue));
}
if (!includeTail && adding.get()) list.get().add(i);
return null;
}))
.filterNonNull();
return stream;
}
public default List toList() {
return asStream().toJavaList();
}
public default FuncList toFuncList() {
return asStream().toImmutableList();
}
public default ImmutableList toImmutableList() {
return asStream().toImmutableList();
}
public default List toMutableList() {
return asStream().toMutableList();
}
public default ArrayList toArrayList() {
return asStream().toArrayList();
}
public default Set toSet() {
return asStream().toSet();
}
//== Plus ==
public default String joining() {
return terminate(stream -> {
return stream()
.mapToObj(StrFuncs::toStr)
.collect(Collectors.joining());
});
}
public default String joining(String delimiter) {
return terminate(stream -> {
return stream()
.mapToObj(StrFuncs::toStr)
.collect(Collectors.joining(delimiter));
});
}
//== Calculate ==
public default T calculate(
DoubleCollectorPlus processor) {
val collected = Collected.ofDouble(processor);
forEach(each -> {
collected.accumulate(each);
});
val value = collected.finish();
return value;
}
public default Tuple2 calculate(
DoubleCollectorPlus processor1,
DoubleCollectorPlus processor2) {
val collected1 = Collected.ofDouble(processor1);
val collected2 = Collected.ofDouble(processor2);
forEach(each -> {
collected1.accumulate(each);
collected2.accumulate(each);
});
val value1 = collected1.finish();
val value2 = collected2.finish();
return Tuple.of(value1, value2);
}
public default Tuple3 calculate(
DoubleCollectorPlus processor1,
DoubleCollectorPlus processor2,
DoubleCollectorPlus processor3) {
val collected1 = Collected.ofDouble(processor1);
val collected2 = Collected.ofDouble(processor2);
val collected3 = Collected.ofDouble(processor3);
forEach(each -> {
collected1.accumulate(each);
collected2.accumulate(each);
collected3.accumulate(each);
});
val value1 = collected1.finish();
val value2 = collected2.finish();
val value3 = collected3.finish();
return Tuple.of(value1, value2, value3);
}
public default Tuple4 calculate(
DoubleCollectorPlus processor1,
DoubleCollectorPlus processor2,
DoubleCollectorPlus processor3,
DoubleCollectorPlus processor4) {
val collected1 = Collected.ofDouble(processor1);
val collected2 = Collected.ofDouble(processor2);
val collected3 = Collected.ofDouble(processor3);
val collected4 = Collected.ofDouble(processor4);
forEach(each -> {
collected1.accumulate(each);
collected2.accumulate(each);
collected3.accumulate(each);
collected4.accumulate(each);
});
val value1 = collected1.finish();
val value2 = collected2.finish();
val value3 = collected3.finish();
val value4 = collected4.finish();
return Tuple.of(value1, value2, value3, value4);
}
public default Tuple5 calculate(
DoubleCollectorPlus processor1,
DoubleCollectorPlus processor2,
DoubleCollectorPlus processor3,
DoubleCollectorPlus processor4,
DoubleCollectorPlus processor5) {
val collected1 = Collected.ofDouble(processor1);
val collected2 = Collected.ofDouble(processor2);
val collected3 = Collected.ofDouble(processor3);
val collected4 = Collected.ofDouble(processor4);
val collected5 = Collected.ofDouble(processor5);
forEach(each -> {
collected1.accumulate(each);
collected2.accumulate(each);
collected3.accumulate(each);
collected4.accumulate(each);
collected5.accumulate(each);
});
val value1 = collected1.finish();
val value2 = collected2.finish();
val value3 = collected3.finish();
val value4 = collected4.finish();
val value5 = collected5.finish();
return Tuple.of(value1, value2, value3, value4, value5);
}
public default Tuple6 calculate(
DoubleCollectorPlus processor1,
DoubleCollectorPlus processor2,
DoubleCollectorPlus processor3,
DoubleCollectorPlus processor4,
DoubleCollectorPlus processor5,
DoubleCollectorPlus processor6) {
val collected1 = Collected.ofDouble(processor1);
val collected2 = Collected.ofDouble(processor2);
val collected3 = Collected.ofDouble(processor3);
val collected4 = Collected.ofDouble(processor4);
val collected5 = Collected.ofDouble(processor5);
val collected6 = Collected.ofDouble(processor6);
forEach(each -> {
collected1.accumulate(each);
collected2.accumulate(each);
collected3.accumulate(each);
collected4.accumulate(each);
collected5.accumulate(each);
collected6.accumulate(each);
});
val value1 = collected1.finish();
val value2 = collected2.finish();
val value3 = collected3.finish();
val value4 = collected4.finish();
val value5 = collected5.finish();
val value6 = collected6.finish();
return Tuple.of(value1, value2, value3, value4, value5, value6);
}
}