All Downloads are FREE. Search and download functionalities are using the official Maven repository.

functionalj.stream.DoubleStreamPlus Maven / Gradle / Ivy

There is a newer version: 1.0.17
Show newest version
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.AtomicLong;
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.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 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 piper) {
        return piper.apply(this);
    }
    
    public default  StreamPlus mapBy(DoubleFunction 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 mapper) {
        StreamPlus stream = StreamPlus.from(stream().mapToObj(mapper));
        stream.onClose(()->{ close(); });
        return stream;
    }
    
    public default  StreamPlus mapToObj(Supplier 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 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));
        });
    }
    
    //== Get ==
    
    public default  T get(DoubleStreamElementProcessor processor) {
        val counter = new AtomicLong(0);
        val iterator = iterator();
        while (iterator.hasNext()) {
            val each  = iterator.nextDouble();
            val index = counter.getAndIncrement();
            processor.processElement(index, each);
        }
        val count = counter.get();
        return processor.processComplete(count);
    }
    
    public default  Tuple2 get(
                DoubleStreamElementProcessor processor1, 
                DoubleStreamElementProcessor processor2) {
        return get(processor1, processor2, Tuple2::of);
    }
    
    public default  T get(
                DoubleStreamElementProcessor processor1, 
                DoubleStreamElementProcessor processor2,
                Func2          combiner) {
        val counter = new AtomicLong(0);
        val iterator = iterator();
        while (iterator.hasNext()) {
            val each  = iterator.nextDouble();
            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(
                DoubleStreamElementProcessor processor1, 
                DoubleStreamElementProcessor processor2, 
                DoubleStreamElementProcessor processor3) {
        return get(processor1, processor2, processor3, Tuple3::of);
    }
    
    public default  T get(
                DoubleStreamElementProcessor processor1, 
                DoubleStreamElementProcessor processor2, 
                DoubleStreamElementProcessor processor3,
                Func3   combiner) {
        val counter = new AtomicLong(0);
        val iterator = iterator();
        while (iterator.hasNext()) {
            val each  = iterator.nextDouble();
            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(
                DoubleStreamElementProcessor processor1, 
                DoubleStreamElementProcessor processor2, 
                DoubleStreamElementProcessor processor3, 
                DoubleStreamElementProcessor processor4) {
        return get(processor1, processor2, processor3, processor4, Tuple4::of);
    }
    
    public default  T get(
                DoubleStreamElementProcessor processor1, 
                DoubleStreamElementProcessor processor2, 
                DoubleStreamElementProcessor processor3,
                DoubleStreamElementProcessor processor4,
                Func4  combiner) {
        val counter = new AtomicLong(0);
        val iterator = iterator();
        while (iterator.hasNext()) {
            val each  = iterator.nextDouble();
            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(
                DoubleStreamElementProcessor processor1, 
                DoubleStreamElementProcessor processor2, 
                DoubleStreamElementProcessor processor3, 
                DoubleStreamElementProcessor processor4, 
                DoubleStreamElementProcessor processor5) {
        return get(processor1, processor2, processor3, processor4, processor5, Tuple5::of);
    }
    
    public default  T get(
                DoubleStreamElementProcessor processor1, 
                DoubleStreamElementProcessor processor2, 
                DoubleStreamElementProcessor processor3,
                DoubleStreamElementProcessor processor4,
                DoubleStreamElementProcessor processor5,
                Func5  combiner) {
        val counter = new AtomicLong(0);
        val iterator = iterator();
        while (iterator.hasNext()) {
            val each  = iterator.nextDouble();
            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(
                DoubleStreamElementProcessor processor1, 
                DoubleStreamElementProcessor processor2, 
                DoubleStreamElementProcessor processor3, 
                DoubleStreamElementProcessor processor4, 
                DoubleStreamElementProcessor processor5, 
                DoubleStreamElementProcessor processor6) {
        return get(processor1, processor2, processor3, processor4, processor5, processor6, Tuple6::of);
    }
    
    public default  T get(
                DoubleStreamElementProcessor processor1, 
                DoubleStreamElementProcessor processor2, 
                DoubleStreamElementProcessor processor3,
                DoubleStreamElementProcessor processor4,
                DoubleStreamElementProcessor processor5,
                DoubleStreamElementProcessor processor6,
                Func6  combiner) {
        val counter = new AtomicLong(0);
        val iterator = iterator();
        while (iterator.hasNext()) {
            val each  = iterator.nextDouble();
            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);
    }
}