blob: 70a9dd9c827bc14e4ebfe37c223aecc9ba48eb98 [file] [log] [blame]
/*
* 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);
}