functionalj.stream.LongStreamPlus Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of functionalj-core Show documentation
Show all versions of functionalj-core Show documentation
The module for FunctionalJ Core.
package functionalj.stream;
import java.util.ArrayList;
import java.util.List;
import java.util.LongSummaryStatistics;
import java.util.OptionalDouble;
import java.util.OptionalLong;
import java.util.PrimitiveIterator;
import java.util.Set;
import java.util.Spliterator;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicLong;
import java.util.concurrent.atomic.AtomicReference;
import java.util.function.BiConsumer;
import java.util.function.Function;
import java.util.function.LongBinaryOperator;
import java.util.function.LongConsumer;
import java.util.function.LongFunction;
import java.util.function.LongPredicate;
import java.util.function.LongSupplier;
import java.util.function.LongToDoubleFunction;
import java.util.function.LongToIntFunction;
import java.util.function.LongUnaryOperator;
import java.util.function.ObjLongConsumer;
import java.util.function.Supplier;
import java.util.stream.Collectors;
import java.util.stream.DoubleStream;
import java.util.stream.LongStream;
import java.util.stream.Stream;
import functionalj.function.Func1;
import functionalj.function.Func2;
import functionalj.function.Func3;
import functionalj.function.Func4;
import functionalj.function.Func5;
import functionalj.function.Func6;
import functionalj.function.FuncUnit1;
import functionalj.functions.StrFuncs;
import functionalj.list.FuncList;
import functionalj.list.ImmutableList;
import functionalj.pipeable.Pipeable;
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 LongStreamPlus extends LongStream {
public static LongStreamPlus of(long ... longs) {
return LongStreamPlus.from(LongStream.of(longs));
}
public static LongStreamPlus from(LongStream longStream) {
return ()->longStream;
}
public static LongStreamPlus concat(LongStreamPlus ... streams) {
return LongStreamPlus.from(StreamPlus.of(streams).flatMap(s -> s.asStream()).mapToLong(i -> i));
}
public static LongStreamPlus empty() {
return LongStreamPlus.from(LongStream.empty());
}
public static LongStreamPlus infinite() {
return LongStreamPlus.from(LongStream.iterate(0, i -> i + 1));
}
public static LongStreamPlus range(int startInclusive, int endExclusive) {
return LongStreamPlus.from(LongStream.range(startInclusive, endExclusive));
}
public static LongStreamPlus rangeClosed(int startInclusive, int endInclusive) {
return LongStreamPlus.from(LongStream.rangeClosed(startInclusive, endInclusive));
}
public static LongStreamPlus generate(LongSupplier s) {
return LongStreamPlus.from(LongStream.generate(s));
}
public static LongStreamPlus iterate(long seed, LongUnaryOperator f) {
return LongStreamPlus.from(LongStream.iterate(seed, f));
}
public static LongStreamPlus compound(long seed, LongUnaryOperator f) {
return LongStreamPlus.from(LongStream.iterate(seed, f));
}
public static LongStreamPlus iterate(long seed1, long seed2, LongBinaryOperator f) {
val counter = new AtomicLong(0);
val int1 = new AtomicLong(seed1);
val int2 = new AtomicLong(seed2);
return LongStreamPlus.generate(()->{
if (counter.getAndIncrement() == 0)
return seed1;
if (counter.getAndIncrement() == 2)
return seed2;
long i2 = int2.get();
long i1 = int1.getAndSet(i2);
long i = f.applyAsLong(i1, i2);
int2.set(i);
return i;
});
}
//== Stream ==
public LongStream 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 LongStreamPlus filter(LongPredicate predicate) {
return LongStreamPlus.from(stream().filter(predicate));
}
@Override
public default LongStreamPlus map(LongUnaryOperator mapper) {
return LongStreamPlus.from(stream().map(mapper));
}
public default Pipeable pipable() {
return Pipeable.of(this);
}
public default T pipe(Function super LongStreamPlus, T> piper) {
return piper.apply(this);
}
public default StreamPlus mapBy(LongFunction extends U> mapper) {
return StreamPlus.from(stream().mapToObj(mapper));
}
public default StreamPlus asStream() {
val stream = StreamPlus.from(stream()
.mapToObj(i -> Long.valueOf(i)));
stream.onClose(()->{ close(); });
return stream;
}
@Override
public default StreamPlus mapToObj(LongFunction 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(LongToIntFunction mapper) {
return IntStreamPlus.from(stream().mapToInt(mapper));
}
@Override
public default DoubleStream mapToDouble(LongToDoubleFunction mapper) {
return stream().mapToDouble (mapper);
}
@Override
public default LongStreamPlus flatMap(LongFunction extends LongStream> mapper) {
return LongStreamPlus.from(stream().flatMap(mapper));
}
public default LongStreamPlus flatMap(Function> mapper) {
return LongStreamPlus.from(stream()
.mapToObj (i -> Long.valueOf(i))
.flatMap (i -> mapper.apply(i))
.mapToLong(i -> i.longValue()));
}
@Override
public default LongStreamPlus distinct() {
return LongStreamPlus.from(stream().distinct());
}
@Override
public default LongStreamPlus sorted() {
return LongStreamPlus.from(stream().sorted());
}
@Override
public default LongStreamPlus peek(LongConsumer action) {
return LongStreamPlus.from(stream().peek(action));
}
@Override
public default LongStreamPlus limit(long maxSize) {
return LongStreamPlus.from(stream().limit(maxSize));
}
@Override
public default LongStreamPlus skip(long n) {
return LongStreamPlus.from(stream().skip(n));
}
@Override
public default void forEach(LongConsumer action) {
terminate(stream-> {
stream
.forEach(action);
});
}
@Override
public default void forEachOrdered(LongConsumer action) {
terminate(stream-> {
stream
.forEachOrdered(action);
});
}
@Override
public default long[] toArray() {
return terminate(stream-> {
return stream
.toArray();
});
}
@Override
public default long reduce(long identity, LongBinaryOperator op) {
return terminate(stream-> {
return stream
.reduce(identity, op);
});
}
@Override
public default OptionalLong reduce(LongBinaryOperator op) {
return terminate(stream-> {
return stream
.reduce(op);
});
}
@Override
public default R collect(Supplier supplier,
ObjLongConsumer accumulator,
BiConsumer combiner) {
return terminate(stream-> {
return stream
.collect(supplier, accumulator, combiner);
});
}
@Override
public default long sum() {
return terminate(stream-> {
return stream
.sum();
});
}
@Override
public default OptionalLong min() {
return terminate(stream-> {
return stream
.min();
});
}
@Override
public default OptionalLong 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 LongSummaryStatistics summaryStatistics() {
return terminate(stream-> {
return stream
.summaryStatistics();
});
}
@Override
public default boolean anyMatch(LongPredicate predicate) {
return terminate(stream-> {
return stream
.anyMatch(predicate);
});
}
@Override
public default boolean allMatch(LongPredicate predicate) {
return terminate(stream-> {
return stream
.allMatch(predicate);
});
}
@Override
public default boolean noneMatch(LongPredicate predicate) {
return terminate(stream-> {
return stream
.noneMatch(predicate);
});
}
@Override
public default OptionalLong findFirst() {
return terminate(stream-> {
return stream
.findFirst();
});
}
@Override
public default OptionalLong findAny() {
return terminate(stream-> {
return stream
.findAny();
});
}
@Override
public default DoubleStreamPlus asDoubleStream() {
return DoubleStreamPlus.from(stream().asDoubleStream());
}
@Override
public default StreamPlus boxed() {
return StreamPlus.from(stream().boxed());
}
@Override
public default LongStreamPlus sequential() {
return LongStreamPlus.from(stream().sequential());
}
@Override
public default LongStreamPlus parallel() {
return LongStreamPlus.from(stream().sequential());
}
@Override
public default PrimitiveIterator.OfLong iterator() {
// TODO - Make sure close is handled properly.
return stream().iterator();
}
@Override
public default Spliterator.OfLong spliterator() {
return stream().spliterator();
}
@Override
public default boolean isParallel() {
return stream().isParallel();
}
@Override
public default LongStreamPlus unordered() {
return LongStreamPlus.from(stream().unordered());
}
@Override
public default LongStreamPlus onClose(Runnable closeHandler) {
return LongStreamPlus.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 AtomicLong(0);
return segment(data -> (index.getAndIncrement() % count) == 0, includeTail);
}
public default StreamPlus segment(LongPredicate startCondition) {
return segment(startCondition, true);
}
public default StreamPlus segment(LongPredicate 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 LongStreamPlus.from(retList.stream().mapToLong(Long::longValue));
}
if (adding.get()) list.get().add(i);
return null;
}))
.filterNonNull();
;
val mainSupplier = (Supplier>)()->mainStream;
if (!includeTail)
return mainStream;
val tailSupplier = (Supplier>)()->{
return StreamPlus.of(
LongStreamPlus.from(
list.get()
.stream()
.mapToLong(Long::longValue)));
};
val resultStream = StreamPlus.of(
mainSupplier,
tailSupplier
)
.flatMap(Supplier::get);
return resultStream;
}
public default StreamPlus segment(LongPredicate startCondition, LongPredicate endCondition) {
return segment(startCondition, endCondition, true);
}
public default StreamPlus segment(LongPredicate startCondition, LongPredicate 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 LongStreamPlus.from(retList.stream().mapToLong(Long::longValue));
}
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));
});
}
//== Get ==
public default T get(LongStreamElementProcessor processor) {
val counter = new AtomicLong(0);
val iterator = iterator();
while (iterator.hasNext()) {
val each = iterator.nextLong();
val index = counter.getAndIncrement();
processor.processElement(index, each);
}
val count = counter.get();
return processor.processComplete(count);
}
public default Tuple2 get(
LongStreamElementProcessor processor1,
LongStreamElementProcessor processor2) {
return get(processor1, processor2, Tuple2::of);
}
public default T get(
LongStreamElementProcessor processor1,
LongStreamElementProcessor processor2,
Func2 combiner) {
val counter = new AtomicLong(0);
val iterator = iterator();
while (iterator.hasNext()) {
val each = iterator.nextLong();
val index = counter.getAndIncrement();
processor1.processElement(index, each);
processor2.processElement(index, each);
}
val count = counter.get();
val value1 = processor1.processComplete(count);
val value2 = processor2.processComplete(count);
return combiner.apply(value1, value2);
}
public default Tuple3 get(
LongStreamElementProcessor processor1,
LongStreamElementProcessor processor2,
LongStreamElementProcessor processor3) {
return get(processor1, processor2, processor3, Tuple3::of);
}
public default T get(
LongStreamElementProcessor processor1,
LongStreamElementProcessor processor2,
LongStreamElementProcessor processor3,
Func3 combiner) {
val counter = new AtomicLong(0);
val iterator = iterator();
while (iterator.hasNext()) {
val each = iterator.nextLong();
val index = counter.getAndIncrement();
processor1.processElement(index, each);
processor2.processElement(index, each);
processor3.processElement(index, each);
}
val count = counter.get();
val value1 = processor1.processComplete(count);
val value2 = processor2.processComplete(count);
val value3 = processor3.processComplete(count);
return combiner.apply(value1, value2, value3);
}
public default Tuple4 get(
LongStreamElementProcessor processor1,
LongStreamElementProcessor processor2,
LongStreamElementProcessor processor3,
LongStreamElementProcessor processor4) {
return get(processor1, processor2, processor3, processor4, Tuple4::of);
}
public default T get(
LongStreamElementProcessor processor1,
LongStreamElementProcessor processor2,
LongStreamElementProcessor processor3,
LongStreamElementProcessor processor4,
Func4 combiner) {
val counter = new AtomicLong(0);
val iterator = iterator();
while (iterator.hasNext()) {
val each = iterator.nextLong();
val index = counter.getAndIncrement();
processor1.processElement(index, each);
processor2.processElement(index, each);
processor3.processElement(index, each);
processor4.processElement(index, each);
}
val count = counter.get();
val value1 = processor1.processComplete(count);
val value2 = processor2.processComplete(count);
val value3 = processor3.processComplete(count);
val value4 = processor4.processComplete(count);
return combiner.apply(value1, value2, value3, value4);
}
public default Tuple5 get(
LongStreamElementProcessor processor1,
LongStreamElementProcessor processor2,
LongStreamElementProcessor processor3,
LongStreamElementProcessor processor4,
LongStreamElementProcessor processor5) {
return get(processor1, processor2, processor3, processor4, processor5, Tuple5::of);
}
public default T get(
LongStreamElementProcessor processor1,
LongStreamElementProcessor processor2,
LongStreamElementProcessor processor3,
LongStreamElementProcessor processor4,
LongStreamElementProcessor processor5,
Func5 combiner) {
val counter = new AtomicLong(0);
val iterator = iterator();
while (iterator.hasNext()) {
val each = iterator.nextLong();
val index = counter.getAndIncrement();
processor1.processElement(index, each);
processor2.processElement(index, each);
processor3.processElement(index, each);
processor4.processElement(index, each);
processor5.processElement(index, each);
}
val count = counter.get();
val value1 = processor1.processComplete(count);
val value2 = processor2.processComplete(count);
val value3 = processor3.processComplete(count);
val value4 = processor4.processComplete(count);
val value5 = processor5.processComplete(count);
return combiner.apply(value1, value2, value3, value4, value5);
}
public default Tuple6 get(
LongStreamElementProcessor processor1,
LongStreamElementProcessor processor2,
LongStreamElementProcessor processor3,
LongStreamElementProcessor processor4,
LongStreamElementProcessor processor5,
LongStreamElementProcessor processor6) {
return get(processor1, processor2, processor3, processor4, processor5, processor6, Tuple6::of);
}
public default T get(
LongStreamElementProcessor processor1,
LongStreamElementProcessor processor2,
LongStreamElementProcessor processor3,
LongStreamElementProcessor processor4,
LongStreamElementProcessor processor5,
LongStreamElementProcessor processor6,
Func6 combiner) {
val counter = new AtomicLong(0);
val iterator = iterator();
while (iterator.hasNext()) {
val each = iterator.nextLong();
val index = counter.getAndIncrement();
processor1.processElement(index, each);
processor2.processElement(index, each);
processor3.processElement(index, each);
processor4.processElement(index, each);
processor5.processElement(index, each);
processor6.processElement(index, each);
}
val count = counter.get();
val value1 = processor1.processComplete(count);
val value2 = processor2.processComplete(count);
val value3 = processor3.processComplete(count);
val value4 = processor4.processComplete(count);
val value5 = processor5.processComplete(count);
val value6 = processor6.processComplete(count);
return combiner.apply(value1, value2, value3, value4, value5, value6);
}
}