com.google.cloud.genomics.dataflow.utils.PairGenerator.java Source code

Java tutorial

Introduction

Here is the source code for com.google.cloud.genomics.dataflow.utils.PairGenerator.java

Source

/*
 * Copyright (C) 2014 Google Inc.
 *
 * 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.google.cloud.genomics.dataflow.utils;

import com.google.cloud.dataflow.sdk.values.KV;
import com.google.common.base.Function;
import com.google.common.collect.ContiguousSet;
import com.google.common.collect.DiscreteDomain;
import com.google.common.collect.FluentIterable;
import com.google.common.collect.Iterators;
import com.google.common.collect.Range;

import java.util.Comparator;
import java.util.Iterator;
import java.util.List;
import java.util.RandomAccess;

/**
 * Generates all combinations (not permutations) of pairs of the elements in the iterable.
 *
 * WITHOUT_REPLACEMENT - generate Nchoose2 combinations WITH_REPLACEMENT- generate Nchoose2
 * combinations plus an additional N pairs (one for each element in the list)
 */
public enum PairGenerator {

    WITH_REPLACEMENT(true), WITHOUT_REPLACEMENT(false);

    private final boolean withReplacement;

    private PairGenerator(boolean withReplacement) {
        this.withReplacement = withReplacement;
    }

    /**
     * * Generates all combinations (not permutations) of pairs of the elements in the iterable.
     *
     * The pairs are often used as keys for results in many n^2 analyses where we end up with
     * (n*(n-1))/2 or ((n*(n-1))/2)+n results.
     *
     * @param list
     * @param comparator - used to enforce an order upon the elements within each pair
     * @return the pairs
     */
    public <X, L extends List<? extends X> & RandomAccess> FluentIterable<KV<X, X>> allPairs(final L list,
            final Comparator<? super X> comparator) {
        return FluentIterable
                .from(ContiguousSet.create(Range.closedOpen(0, list.size()), DiscreteDomain.integers()))
                .transformAndConcat(new Function<Integer, Iterable<KV<X, X>>>() {
                    @Override
                    public Iterable<KV<X, X>> apply(final Integer i) {
                        return new Iterable<KV<X, X>>() {
                            @Override
                            public Iterator<KV<X, X>> iterator() {
                                int iteratorIndex = i;
                                if (!withReplacement) {
                                    iteratorIndex++;
                                }
                                return Iterators.transform(list.listIterator(iteratorIndex),
                                        new Function<X, KV<X, X>>() {

                                            private final X key = list.get(i);

                                            @Override
                                            public KV<X, X> apply(X value) {
                                                boolean swap = 0 < comparator.compare(key, value);
                                                return KV.of(swap ? value : key, swap ? key : value);
                                            }
                                        });
                            }
                        };
                    }
                });
    }
}