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

org.infinispan.commons.util.FlattenSpliterator Maven / Gradle / Ivy

There is a newer version: 15.1.0.Dev04
Show newest version
package org.infinispan.commons.util;

import java.util.Collection;
import java.util.Spliterator;
import java.util.function.Consumer;
import java.util.function.IntFunction;

/**
 * Composes an array of Collections into a spliterator. This spliterator will only split up to the collection and will
 * not split the spliterator from the collection itself.
 * @author wburns
 * @since 9.3
 */
public class FlattenSpliterator implements Spliterator {
   private final IntFunction> toCollection;
   private final int length;
   private int index;        // current index, modified on advance/split
   private final int fence;  // one past last index
   private final int characteristics;

   private Spliterator currentSpliterator;

   public FlattenSpliterator(IntFunction> toCollection, int length, int additionalCharacteristics) {
      this(toCollection, length, 0, length, additionalCharacteristics);
   }

   private FlattenSpliterator(IntFunction> toCollection, int length, int index, int fence, int characteristics) {
      this.toCollection = toCollection;
      this.length = length;
      if (index < 0) {
         throw new IllegalArgumentException("Index " + index + " was less than 0!");
      }
      this.index = index;
      this.fence = fence;
      this.characteristics = characteristics;
   }

   @Override
   public boolean tryAdvance(Consumer action) {
      boolean advanced = false;
      // If the current spliterator is null or can't advance the current action try the next one
      while ((currentSpliterator == null || !(advanced = currentSpliterator.tryAdvance(action))) && index < fence) {
         currentSpliterator = toCollection.apply(index++).spliterator();
      }
      return advanced;
   }

   @Override
   public void forEachRemaining(Consumer action) {
      for (; index < fence; ++index) {
         toCollection.apply(index).spliterator().forEachRemaining(action);
      }
   }

   @Override
   public Spliterator trySplit() {
      int lo = index, mid = (lo + fence) >>> 1;
      return (lo >= mid)
            ? null
            : new FlattenSpliterator<>(toCollection, length, lo, index = mid, characteristics);
   }

   @Override
   public long estimateSize() {
      long estimate = 0;
      if (currentSpliterator != null) {
         estimate += currentSpliterator.estimateSize();
      }
      if (estimate == Long.MAX_VALUE) {
         return estimate;
      }
      for (int i = index; i < fence; ++i) {
         estimate += toCollection.apply(i).size();
         if (estimate < 0) {
            return Long.MAX_VALUE;
         }
      }
      return estimate;
   }

   @Override
   public int characteristics() {
      return characteristics;
   }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy