| /* |
| * Copyright 2016 Google Inc. |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); you may not |
| * use this file except in compliance with the License. You may obtain a copy of |
| * the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT |
| * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the |
| * License for the specific language governing permissions and limitations under |
| * the License. |
| */ |
| package java.util.stream; |
| |
| import static javaemul.internal.InternalPreconditions.checkState; |
| |
| import java.util.Arrays; |
| import java.util.Collections; |
| import java.util.Comparator; |
| import java.util.Optional; |
| import java.util.Spliterator; |
| import java.util.Spliterators; |
| import java.util.Spliterators.AbstractSpliterator; |
| import java.util.function.BiConsumer; |
| import java.util.function.BiFunction; |
| import java.util.function.BinaryOperator; |
| import java.util.function.Consumer; |
| import java.util.function.Function; |
| import java.util.function.IntFunction; |
| import java.util.function.Predicate; |
| import java.util.function.Supplier; |
| import java.util.function.ToDoubleFunction; |
| import java.util.function.ToIntFunction; |
| import java.util.function.ToLongFunction; |
| import java.util.function.UnaryOperator; |
| |
| /** |
| * See <a href="https://docs.oracle.com/javase/8/docs/api/java/util/stream/Stream.html"> |
| * the official Java API doc</a> for details. |
| * |
| * @param <T> the type of data being streamed |
| */ |
| public interface Stream<T> extends BaseStream<T, Stream<T>> { |
| |
| /** |
| * See <a href="https://docs.oracle.com/javase/8/docs/api/java/util/stream/Stream.Builder.html"> |
| * the official Java API doc</a> for details. |
| */ |
| interface Builder<T> extends Consumer<T> { |
| @Override |
| void accept(T t); |
| |
| default Stream.Builder<T> add(T t) { |
| accept(t); |
| return this; |
| } |
| |
| Stream<T> build(); |
| } |
| |
| static <T> Stream.Builder<T> builder() { |
| return new Builder<T>() { |
| private Object[] items = new Object[0]; |
| |
| @Override |
| public void accept(T t) { |
| checkState(items != null, "Builder already built"); |
| items[items.length] = t; |
| } |
| |
| @Override |
| @SuppressWarnings("unchecked") |
| public Stream<T> build() { |
| checkState(items != null, "Builder already built"); |
| Stream<T> stream = (Stream<T>) Arrays.stream(items); |
| items = null; |
| return stream; |
| } |
| }; |
| } |
| |
| static <T> Stream<T> concat(Stream<? extends T> a, Stream<? extends T> b) { |
| // This is nearly the same as flatMap, but inlined, wrapped around a single spliterator of |
| // these two objects, and without close() called as the stream progresses. Instead, close is |
| // invoked as part of the resulting stream's own onClose, so that either can fail without |
| // affecting the other, and correctly collecting suppressed exceptions. |
| |
| // TODO replace this flatMap-ish spliterator with one that directly combines the two root |
| // streams |
| Spliterator<? extends Stream<? extends T>> spliteratorOfStreams = |
| Arrays.asList(a, b).spliterator(); |
| |
| AbstractSpliterator<T> spliterator = |
| new Spliterators.AbstractSpliterator<T>(Long.MAX_VALUE, 0) { |
| Spliterator<? extends T> next; |
| |
| @Override |
| public boolean tryAdvance(Consumer<? super T> action) { |
| // look for a new spliterator |
| while (advanceToNextSpliterator()) { |
| // if we have one, try to read and use it |
| if (next.tryAdvance(action)) { |
| return true; |
| } else { |
| // failed, null it out so we can find another |
| next = null; |
| } |
| } |
| return false; |
| } |
| |
| private boolean advanceToNextSpliterator() { |
| while (next == null) { |
| if (!spliteratorOfStreams.tryAdvance( |
| n -> { |
| if (n != null) { |
| next = n.spliterator(); |
| } |
| })) { |
| return false; |
| } |
| } |
| return true; |
| } |
| }; |
| |
| Stream<T> result = new StreamImpl<T>(null, spliterator); |
| |
| return result.onClose(a::close).onClose(b::close); |
| } |
| |
| static <T> Stream<T> empty() { |
| return new StreamImpl.Empty<T>(null); |
| } |
| |
| static <T> Stream<T> generate(Supplier<T> s) { |
| AbstractSpliterator<T> spliterator = |
| new Spliterators.AbstractSpliterator<T>( |
| Long.MAX_VALUE, Spliterator.IMMUTABLE | Spliterator.ORDERED) { |
| @Override |
| public boolean tryAdvance(Consumer<? super T> action) { |
| action.accept(s.get()); |
| return true; |
| } |
| }; |
| return StreamSupport.stream(spliterator, false); |
| } |
| |
| static <T> Stream<T> iterate(T seed, UnaryOperator<T> f) { |
| AbstractSpliterator<T> spliterator = |
| new Spliterators.AbstractSpliterator<T>( |
| Long.MAX_VALUE, Spliterator.IMMUTABLE | Spliterator.ORDERED) { |
| private T next = seed; |
| |
| @Override |
| public boolean tryAdvance(Consumer<? super T> action) { |
| action.accept(next); |
| next = f.apply(next); |
| return true; |
| } |
| }; |
| return StreamSupport.stream(spliterator, false); |
| } |
| |
| static <T> Stream<T> of(T t) { |
| // TODO consider a splittable that returns only a single value, either for use here or in the |
| // singleton collection types |
| return Collections.singleton(t).stream(); |
| } |
| |
| static <T> Stream<T> of(T... values) { |
| return Arrays.stream(values); |
| } |
| |
| boolean allMatch(Predicate<? super T> predicate); |
| |
| boolean anyMatch(Predicate<? super T> predicate); |
| |
| <R, A> R collect(Collector<? super T, A, R> collector); |
| |
| <R> R collect( |
| Supplier<R> supplier, BiConsumer<R, ? super T> accumulator, BiConsumer<R, R> combiner); |
| |
| long count(); |
| |
| Stream<T> distinct(); |
| |
| Stream<T> filter(Predicate<? super T> predicate); |
| |
| Optional<T> findAny(); |
| |
| Optional<T> findFirst(); |
| |
| <R> Stream<R> flatMap(Function<? super T, ? extends Stream<? extends R>> mapper); |
| |
| DoubleStream flatMapToDouble(Function<? super T, ? extends DoubleStream> mapper); |
| |
| IntStream flatMapToInt(Function<? super T, ? extends IntStream> mapper); |
| |
| LongStream flatMapToLong(Function<? super T, ? extends LongStream> mapper); |
| |
| void forEach(Consumer<? super T> action); |
| |
| void forEachOrdered(Consumer<? super T> action); |
| |
| Stream<T> limit(long maxSize); |
| |
| <R> Stream<R> map(Function<? super T, ? extends R> mapper); |
| |
| DoubleStream mapToDouble(ToDoubleFunction<? super T> mapper); |
| |
| IntStream mapToInt(ToIntFunction<? super T> mapper); |
| |
| LongStream mapToLong(ToLongFunction<? super T> mapper); |
| |
| Optional<T> max(Comparator<? super T> comparator); |
| |
| Optional<T> min(Comparator<? super T> comparator); |
| |
| boolean noneMatch(Predicate<? super T> predicate); |
| |
| Stream<T> peek(Consumer<? super T> action); |
| |
| Optional<T> reduce(BinaryOperator<T> accumulator); |
| |
| T reduce(T identity, BinaryOperator<T> accumulator); |
| |
| <U> U reduce(U identity, BiFunction<U, ? super T, U> accumulator, BinaryOperator<U> combiner); |
| |
| Stream<T> skip(long n); |
| |
| Stream<T> sorted(); |
| |
| Stream<T> sorted(Comparator<? super T> comparator); |
| |
| Object[] toArray(); |
| |
| <A> A[] toArray(IntFunction<A[]> generator); |
| } |