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

com.palantir.common.collect.IteratorUtils Maven / Gradle / Ivy

There is a newer version: 0.1152.0
Show newest version
/*
 * (c) Copyright 2018 Palantir Technologies Inc. All rights reserved.
 *
 * 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 com.palantir.common.collect;

import com.google.common.base.Function;
import com.google.common.base.Predicate;
import com.google.common.collect.AbstractIterator;
import com.google.common.collect.ForwardingIterator;
import com.google.common.collect.Iterators;
import com.google.common.collect.PeekingIterator;
import com.google.common.collect.UnmodifiableIterator;
import com.palantir.common.base.Throwables;
import com.palantir.common.base.Visitors;
import com.palantir.common.visitor.Visitor;
import com.palantir.logsafe.Preconditions;
import com.palantir.logsafe.exceptions.SafeIllegalStateException;
import com.palantir.util.Pair;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Comparator;
import java.util.Iterator;
import java.util.List;
import java.util.Random;
import javax.annotation.Nullable;

public final class IteratorUtils {

    private static final Random rand = new Random();

    /**
     * Get first element or null if the iterator is empty.
     */
    public static  T getFirst(Iterator iter) {
        return getFirst(iter, null);
    }

    public static  T getFirst(Iterator iter, @Nullable T defaultValue) {
        if (iter.hasNext()) {
            return iter.next();
        } else {
            return defaultValue;
        }
    }

    public static  Collection chooseRandomElements(Iterator it, final int k) {
        Preconditions.checkArgument(k >= 0);
        List ret = new ArrayList<>();

        int i = 0; // i is the element number
        while (it.hasNext()) {
            i++;
            T next = it.next();
            if (ret.size() < k) {
                ret.add(next);
            } else {
                int nextInt = rand.nextInt(i);
                if (nextInt < k) {
                    ret.set(nextInt, next);
                }
            }
            Preconditions.checkState(i != Integer.MAX_VALUE, "i is too big");
        }
        return ret;
    }

    /**
     * Returns the elements of {@code unfiltered} that satisfy a predicate,
     * calling {@code passVisitor.visit()} on all that satisfy the predicate and
     * {@code failVisitor.visit()} on all that do not.
     */
    public static  UnmodifiableIterator filterAndVisit(
            final Iterator unfiltered,
            final Predicate predicate,
            @Nullable Visitor passVisitor,
            @Nullable Visitor failVisitor) {
        Preconditions.checkNotNull(unfiltered);
        Preconditions.checkNotNull(predicate);
        final Visitor actualPassVisitor;
        if (passVisitor != null) {
            actualPassVisitor = passVisitor;
        } else {
            actualPassVisitor = Visitors.emptyVisitor();
        }
        final Visitor actualFailVisitor;
        if (failVisitor != null) {
            actualFailVisitor = failVisitor;
        } else {
            actualFailVisitor = Visitors.emptyVisitor();
        }
        return new AbstractIterator() {
            @Override
            protected T computeNext() {
                try {
                    while (unfiltered.hasNext()) {
                        T element = unfiltered.next();
                        if (predicate.apply(element)) {
                            actualPassVisitor.visit(element);
                            return element;
                        } else {
                            actualFailVisitor.visit(element);
                        }
                    }
                    return endOfData();
                } catch (Exception e) {
                    throw Throwables.throwUncheckedException(e);
                }
            }
        };
    }

    public static  Iterator wrap(final Iterator iterator) {
        if (iterator == null) {
            return null;
        }
        return new ForwardingIterator() {
            @SuppressWarnings("unchecked")
            @Override
            protected Iterator delegate() {
                return (Iterator) iterator;
            }
        };
    }

    private IteratorUtils() {
        throw new SafeIllegalStateException("uninstantiable");
    }

    /**
     * The iterators provided to this function have to be sorted and strictly increasing.
     */
    @SuppressWarnings("BadAssert") // performance sensitive assert checks
    public static  Iterator mergeIterators(
            Iterator one,
            Iterator two,
            final Comparator ordering,
            final Function, ? extends T> mergeFunction) {
        Preconditions.checkNotNull(mergeFunction);
        Preconditions.checkNotNull(ordering);
        final PeekingIterator a = Iterators.peekingIterator(one);
        final PeekingIterator b = Iterators.peekingIterator(two);
        return new AbstractIterator() {
            @Override
            protected T computeNext() {
                if (!a.hasNext() && !b.hasNext()) {
                    return endOfData();
                }
                if (!a.hasNext()) {
                    T ret = b.next();
                    assert !b.hasNext() || ordering.compare(ret, b.peek()) < 0;
                    return ret;
                }
                if (!b.hasNext()) {
                    T ret = a.next();
                    assert !a.hasNext() || ordering.compare(ret, a.peek()) < 0;
                    return ret;
                }
                T peekA = a.peek();
                T peekB = b.peek();
                int comp = ordering.compare(peekA, peekB);
                if (comp == 0) {
                    return mergeFunction.apply(Pair.create(a.next(), b.next()));
                } else if (comp < 0) {
                    T ret = a.next();
                    assert !a.hasNext() || ordering.compare(ret, a.peek()) < 0;
                    return ret;
                } else {
                    T ret = b.next();
                    assert !b.hasNext() || ordering.compare(ret, b.peek()) < 0;
                    return ret;
                }
            }
        };
    }

    public static  Iterator> zip(final Iterator it1, final Iterator it2) {
        return new AbstractIterator>() {
            @Override
            protected Pair computeNext() {
                if (!it1.hasNext() && !it2.hasNext()) {
                    return endOfData();
                }
                T v1 = null;
                if (it1.hasNext()) {
                    v1 = it1.next();
                }
                U v2 = null;
                if (it2.hasNext()) {
                    v2 = it2.next();
                }
                return Pair.create(v1, v2);
            }
        };
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy