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

net.digitalid.utility.functional.iterators.FlatteningIterator Maven / Gradle / Ivy

The newest version!
package net.digitalid.utility.functional.iterators;

import java.util.Collection;
import java.util.Iterator;
import java.util.NoSuchElementException;

import javax.annotation.Nonnull;
import javax.annotation.Nullable;

import net.digitalid.utility.annotations.generics.Specifiable;
import net.digitalid.utility.annotations.method.Impure;
import net.digitalid.utility.annotations.method.Pure;
import net.digitalid.utility.annotations.ownership.Capturable;
import net.digitalid.utility.annotations.ownership.Captured;
import net.digitalid.utility.annotations.ownership.NonCapturable;
import net.digitalid.utility.functional.iterables.FiniteIterable;
import net.digitalid.utility.validation.annotations.math.NonNegative;
import net.digitalid.utility.validation.annotations.type.Mutable;

/**
 * This class implements a flattening iterator that iterates over the elements of the given iterator with all collections up to the given level flattened.
 */
@Mutable
public class FlatteningIterator<@Specifiable OUTPUT, @Specifiable INPUT> extends SingleIteratorBasedIterator {
    
    /* -------------------------------------------------- Level -------------------------------------------------- */
    
    protected final @NonNegative int level;
    
    /* -------------------------------------------------- Constructors -------------------------------------------------- */
    
    protected FlatteningIterator(@Captured @Nonnull Iterator primaryIterator, @NonNegative int level) {
        super(primaryIterator);
        
        this.level = level;
    }
    
    /**
     * Returns a new flattening iterator that iterates over the elements of the given iterator with all collections up to the given level flattened.
     */
    @Pure
    public static @Capturable <@Specifiable OUTPUT, @Specifiable INPUT> @Nonnull FlatteningIterator with(@Captured @Nonnull Iterator iterator, @NonNegative int level) {
        return new FlatteningIterator<>(iterator, level);
    }
    
    /* -------------------------------------------------- Methods -------------------------------------------------- */
    
    private @Nullable INPUT nextElement = null;
    
    private boolean found = false;
    
    private @Nullable Iterator subiterator = null;
    
    @Pure
    @Override
    public boolean hasNext() {
        if (subiterator != null) {
            if (subiterator.hasNext()) {
                return true;
            } else {
                subiterator = null;
            }
        }
        
        assert subiterator == null;
        
        if (found) {
            return true;
        } else {
            while (primaryIterator.hasNext()) {
                final INPUT element = primaryIterator.next();
                if (level > 0) {
                    final FiniteIterable iterable;
                    if (element instanceof Collection) { iterable = FiniteIterable.of((Collection) element); }
                    else if (element instanceof Object[]) { iterable = FiniteIterable.of((Object[]) element); }
                    else { iterable = null; }
                    if (iterable != null) {
                        subiterator = new FlatteningIterator<>(iterable.iterator(), level - 1);
                        if (subiterator.hasNext()) {
                            return true;
                        } else {
                            subiterator = null;
                            continue;
                        }
                    }
                }
                nextElement = element;
                found = true;
                return true;
            }
            return false;
        }
    }
    
    @Impure
    @Override
    @SuppressWarnings("unchecked")
    public @NonCapturable OUTPUT next() {
        if (hasNext()) {
            if (subiterator != null) {
                return subiterator.next();
            } else {
                found = false;
                return (OUTPUT) nextElement;
            }
        } else {
            throw new NoSuchElementException();
        }
    }
    
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy