|  | // Copyright 2018 The Bazel Authors. 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.google.devtools.build.lib.collect; | 
|  |  | 
|  | import com.google.common.base.Preconditions; | 
|  | import com.google.common.collect.ImmutableList; | 
|  | import java.util.Comparator; | 
|  | import java.util.PriorityQueue; | 
|  |  | 
|  | /** | 
|  | * A streaming aggregator that, given a {@code k}, allows a streaming aggregation of {@code n} | 
|  | * elements into the {@code min(k, n)} most extreme, in {@code O(min(k, n))} memory and {@code O(n * | 
|  | * log(min(k, n)))} time. | 
|  | */ | 
|  | public abstract class Extrema<T> { | 
|  | private static final Extrema<Object> EMPTY = new EmptyExtrema<>(); | 
|  |  | 
|  | /** | 
|  | * Creates an {@link Extrema} that can aggregate {@code n} elements into the {@code min(k, n)} | 
|  | * smallest. | 
|  | */ | 
|  | public static <T extends Comparable<T>> Extrema<T> min(int k) { | 
|  | return min(k, Comparator.<T>naturalOrder()); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Creates an {@link Extrema} that can aggregate {@code n} elements into the {@code min(k, n)} | 
|  | * smallest, per the given {@code comparator}. | 
|  | */ | 
|  | public static <T> Extrema<T> min(int k, Comparator<T> comparator) { | 
|  | return create(k, comparator); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Creates an {@link Extrema} that can aggregate {@code n} elements into the {@code min(k, n)} | 
|  | * largest. | 
|  | */ | 
|  | public static <T extends Comparable<T>> Extrema<T> max(int k) { | 
|  | return max(k, Comparator.<T>naturalOrder()); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Creates an {@link Extrema} that can aggregate {@code n} elements into the {@code min(k, n)} | 
|  | * largest, per the given {@code comparator}. | 
|  | */ | 
|  | public static <T> Extrema<T> max(int k, Comparator<T> comparator) { | 
|  | return create(k, comparator.reversed()); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Aggregates the given element. | 
|  | * | 
|  | * <p>See {@link #getExtremeElements()}. | 
|  | */ | 
|  | public abstract void aggregate(T element); | 
|  |  | 
|  | /** | 
|  | * For an {@link Extrema} created with {@code k} and with {@code n} calls to {@link #aggregate} | 
|  | * since the most recent call to {@link #clear}, returns the {@code min(k, n)} most extreme of the | 
|  | * those elements, sorted from most extreme to least extreme. | 
|  | */ | 
|  | public abstract ImmutableList<T> getExtremeElements(); | 
|  |  | 
|  | /** Returns true iff {@link #getExtremeElements()} would return an empty result. */ | 
|  | public abstract boolean isEmpty(); | 
|  |  | 
|  | /** | 
|  | * Disregards all the elements {@link #aggregate}'ed already. | 
|  | * | 
|  | * <p>See {@link #getExtremeElements()}. | 
|  | */ | 
|  | public abstract void clear(); | 
|  |  | 
|  | @SuppressWarnings("unchecked") | 
|  | private static <T> Extrema<T> create(int k, Comparator<T> comparator) { | 
|  | Preconditions.checkArgument(k >= 0, "invalid k (%s), must be >=0", k); | 
|  | return k == 0 ? (Extrema<T>) EMPTY : new RegularExtrema<>(k, comparator); | 
|  | } | 
|  |  | 
|  | private static class EmptyExtrema<T> extends Extrema<T> { | 
|  |  | 
|  | @Override | 
|  | public void aggregate(T element) { | 
|  | // no-op. | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public ImmutableList<T> getExtremeElements() { | 
|  | return ImmutableList.of(); | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public void clear() { | 
|  | // no-op. | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public boolean isEmpty() { | 
|  | return true; | 
|  | } | 
|  | } | 
|  |  | 
|  | private static class RegularExtrema<T> extends Extrema<T> { | 
|  | private final int k; | 
|  | private final Comparator<T> extremaComparator; | 
|  | private final PriorityQueue<T> priorityQueue; | 
|  |  | 
|  | /** | 
|  | * @param k the number of extreme elements to compute | 
|  | * @param extremaComparator a comparator such that {@code extremaComparator(a, b) < 0} iff | 
|  | *     {@code a} is more extreme than {@code b} | 
|  | */ | 
|  | private RegularExtrema(int k, Comparator<T> extremaComparator) { | 
|  | this.k = k; | 
|  | this.extremaComparator = extremaComparator; | 
|  | this.priorityQueue = | 
|  | new PriorityQueue<>( | 
|  | /*initialCapacity=*/ k, | 
|  | // Our implementation strategy is to keep a priority queue of the k most extreme | 
|  | // elements | 
|  | // encountered, ordered backwards; this way we have constant-time access to the least | 
|  | // extreme among these elements. | 
|  | extremaComparator.reversed()); | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public void aggregate(T element) { | 
|  | if (priorityQueue.size() < k) { | 
|  | priorityQueue.add(element); | 
|  | } else { | 
|  | if (extremaComparator.compare(element, priorityQueue.peek()) < 0) { | 
|  | // Suppose the least extreme of the current k most extreme elements is e. If the new | 
|  | // element | 
|  | // is more extreme than e, then (i) it must be among the new k most extreme among the (2) | 
|  | // e | 
|  | // must not be. | 
|  | priorityQueue.remove(); | 
|  | priorityQueue.add(element); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public ImmutableList<T> getExtremeElements() { | 
|  | return ImmutableList.sortedCopyOf(extremaComparator, priorityQueue); | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public void clear() { | 
|  | priorityQueue.clear(); | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public boolean isEmpty() { | 
|  | return priorityQueue.isEmpty(); | 
|  | } | 
|  | } | 
|  | } |