com.twitter.finagle.loadbalancer.Heap.scala Maven / Gradle / Ivy
The newest version!
package com.twitter.finagle.loadbalancer
import scala.annotation.tailrec
private[loadbalancer] object Heap {
trait Indexer[-T] {
def apply(t: T, i: Int)
}
object Indexer {
object nil extends Indexer[Any] {
def apply(t: Any, i: Int) {}
}
}
def apply[T](ord: Ordering[T], indexer: Indexer[T] = Indexer.nil) =
new Heap(ord, indexer)
}
/**
* Provide heap operations for some type T.
*
* nb: 1-indexed heaps, ranges are inclusive.
* indexer will only be invoked on swaps. the
* caller is responsible for maintaining the
* initial value.
*/
private[loadbalancer] class Heap[T](
ord: Ordering[T], indexer: Heap.Indexer[T]
) {
import ord._
def swap(heap: Array[T], i: Int, j: Int) {
val tmp = heap(i)
heap(i) = heap(j)
heap(j) = tmp
indexer(heap(i), i)
indexer(heap(j), j)
}
@tailrec
final def fixDown(heap: Array[T], i: Int, j: Int) {
if (j < i*2) return
val m = if (j == i*2 || heap(2*i) < heap(2*i+1)) 2*i else 2*i + 1
if (heap(m) < heap(i)) {
swap(heap, i, m)
fixDown(heap, m, j)
}
}
@tailrec
final def fixUp(heap: Array[T], i: Int) {
if (i != 1 && heap(i) < heap(i/2)) {
swap(heap, i, i/2)
fixUp(heap, i/2)
}
}
def isValid(heap: Array[T], i: Int, j: Int): Boolean =
if (j < i*2) true else {
val left = heap(i) < heap(i*2)
val right = if (j == i*2) true else heap(i) < heap(i*2+1)
left && right
}
}
© 2015 - 2024 Weber Informatics LLC | Privacy Policy