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

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