public class BinaryHeap<E> implements Heap<E> {
private static final int DEFAULT_CAPACITY = 16;
private int size;
private E[] elements;
private Comparator<E> comparator;
public BinaryHeap() {
this(null, null);
}
public BinaryHeap(Comparator<E> comparator) {
this(null, comparator);
}
public BinaryHeap(E[] elements) {
this(elements, null);
}
public BinaryHeap(E[] elements, Comparator<E> comparator) {
this.comparator = comparator;
if (elements == null || elements.length == 0) {
this.elements = (E[])new Object[DEFAULT_CAPACITY];
} else {
int capacity = Math.max(DEFAULT_CAPACITY, elements.length);
this.elements = (E[])new Object[capacity];
for (int i = 0; i < elements.length; i++) {
this.elements[i] = elements[i];
}
size = elements.length;
heapify();
}
}
@Override
public int size() {
return size;
}
@Override
public boolean isEmpty() {
return size == 0;
}
@Override
public void clear() {
for (int i = 0; i < size; i++) {
elements[i] = null;
}
size = 0;
}
@Override
public void add(E element) {
elementNotNullCheck(element);
ensureCapacity(size + 1);
elements[size++] = element;
siftUp(size - 1);
}
@Override
public E get() {
emptyCheck();
return elements[0];
}
@Override
public E remove() {
emptyCheck();
int lastIndex = --size;
E root = elements[0];
elements[0] = elements[lastIndex];
elements[lastIndex] = null;
siftDown(0);
return root;
}
@Override
public E replace(E element) {
elementNotNullCheck(element);
E root = null;
if (size == 0) {
elements[size++] = element;
} else {
root = elements[0];
elements[0] = element;
siftDown(0);
}
return root;
}
/**
* 批量建堆, 采用自下而上的下滤
*/
private void heapify() {
for (int i = (size >> 1) - 1; i >= 0; i--) {
siftDown(i);
}
}
/**
* 上滤操作
* @param index
*/
private void siftUp(int index) {
E element = elements[index];
while (index > 0) {
int parentIndex = (index - 1) >> 1;
E parent = elements[parentIndex];
if (compare(element, parent) <= 0) break;
elements[index] = parent;
index = parentIndex;
}
elements[index] = element;
}
/**
* 下滤操作
* @param index
*/
private void siftDown(int index) {
E element = elements[index];
int half = size >> 1; // 非叶子节点数量(第一个叶子节点的索引)
while (index < half) {
int childIndex = (index << 1) + 1;
E child = elements[childIndex];
int rightIndex = childIndex + 1;
if (rightIndex < size && compare(elements[rightIndex], child) > 0) {
child = elements[childIndex = rightIndex];
}
if (compare(element, child) >= 0) break;
elements[index] = child;
index = childIndex;
}
elements[index] = element;
}
private int compare(E e1, E e2) {
if (comparator != null) return comparator.compare(e1, e2);
return ((Comparable)e1).compareTo(e2);
}
private void ensureCapacity(int capacity) {
int oldCapacity = elements.length;
if (oldCapacity >= capacity) return;
E[] newElements = (E[]) new Object[oldCapacity + (oldCapacity >> 1)];
for (int i = 0; i < size; i++) {
newElements[i] = elements[i];
}
elements = newElements;
}
private void emptyCheck() {
if (size == 0) throw new IndexOutOfBoundsException("Heap is empty");
}
private void elementNotNullCheck(E element) {
if (element == null) throw new NullPointerException("Element must not be null");
}
}
BinaryHeap
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。