com.adgear.anoa.read.LookAheadIterator Maven / Gradle / Ivy
package com.adgear.anoa.read;
import java.io.Closeable;
import java.io.IOException;
import java.io.UncheckedIOException;
import java.util.Iterator;
import java.util.NoSuchElementException;
import java.util.Spliterator;
import java.util.Spliterators;
import java.util.function.Consumer;
import java.util.function.Function;
import java.util.function.Supplier;
import java.util.function.UnaryOperator;
import java.util.stream.Stream;
import java.util.stream.StreamSupport;
/**
* A generic {@link Iterator} implementation which performs one-step look-ahead.
*
* @param Value type
*/
final public class LookAheadIterator implements Iterator {
final private Supplier noNext;
final private UnaryOperator next;
private long counter = 0;
private Closeable closeable;
private boolean isStale;
private boolean hasNext;
private T nextValue;
/**
* @param noNext called in hasNext(), will cause it to return false if itself returns true.
* @param nextFactory a function which builds a function which returns the next element based on
* the previous one, and can trigger the end of the iteration as a
* side-effect.
* @param closeable {@link Closeable#close()} will be called at the end of the iteration
*/
public LookAheadIterator(
Supplier noNext,
Function, UnaryOperator> nextFactory,
Closeable closeable) {
this.next = nextFactory.apply(this::setHasNext);
this.noNext = noNext;
this.closeable = closeable;
reset(null);
}
/**
* Returns a sequential stream wrapping the generated {@code LookAheadIterator} instance.
*/
static public Stream stream(
Supplier noNext,
Function, UnaryOperator> nextFactory,
Closeable closeable) {
return new LookAheadIterator<>(noNext, nextFactory, closeable).asStream();
}
/**
* Returns a sequential stream wrapping the generated {@code LookAheadIterator} instance.
*/
static public Stream stream(
Supplier noNext,
Function, UnaryOperator> nextFactory) {
return stream(noNext, nextFactory, () -> {
});
}
void reset(T nextValue) {
this.isStale = true;
this.hasNext = true;
this.nextValue = nextValue;
}
protected void setHasNext(boolean hasNext) {
this.hasNext = hasNext;
}
@Override
public boolean hasNext() {
if (isStale) {
isStale = false;
if (noNext.get()) {
setHasNext(false);
} else {
++counter;
nextValue = next.apply(nextValue);
}
}
if (!hasNext) {
try {
closeable.close();
} catch (IOException e) {
throw new UncheckedIOException(e);
}
closeable = null;
}
return hasNext;
}
@Override
public T next() {
if (!hasNext()) {
throw new NoSuchElementException();
}
isStale = true;
return nextValue;
}
Spliterator asSpliterator() {
return Spliterators.spliteratorUnknownSize(this, Spliterator.NONNULL | Spliterator.ORDERED);
}
Stream asStream() {
return StreamSupport.stream(asSpliterator(), false);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy