堆排序原理以及实现
堆性质的简介
堆的创建
public void maxHeap(int[] A,int i,int length){
int left = i*2 + 1; // 根结点的左孩子
int right = i*2 + 2; // 根结点的右孩子
int largest = i;
// 如果存在左孩子,且左孩子大于插入结点的值
if(left < length && A[i] < A[left]){
largest = left;
}
// 如果存在右孩子,且右孩子大于插入结点的值
if(right < length && A[largest] < A[right]){
largest = right;
}
// 如果最大值不是当前插入的结点的值,交换两个结点之间的值
if(largest != i){
exchange(A, i, largest);
maxHeap(A,largest,length);
}
}
private void exchange(int[] A, int i, int largest) {
A[i] = A[i] ^ A[largest];
A[largest] = A[i] ^ A[largest];
A[i] = A[i] ^ A[largest];
}
从程序中我们可以看出,对于新插入的结点在数组的第一个位置,首先比较其对应的左孩子与右孩子的值,将三个结点中值最大的设置为根结点,将要插入的结点与其交换位置。利用下标 largest,i 判断是否发生过交换,如果发生过交换,交换对应两个结点的值,利用递归的形式重复上面的步骤,未发生交换则退出。下图为交换的过程。
public void buildMaxHeap(int[] A){
for(int i = A.length >> 1; i >= 0 ; i--){
maxHeap(A,i,A.length);
}
}
通过上面的几个程序,我们就可以建立一个堆了。利用创建好的堆进行堆排序
// 堆排序
public void heapSort(){
// 考虑到堆是完全二叉树,所以可以考虑使用连续的数组存储二叉树的结点。
int a[] = {4,1,3,2,16,9,10,14,8,7};
// 建立最大堆
buildMaxHeap(a);
System.out.println(Arrays.toString(a));
// 堆排序
int length = a.length;
for(int i = 0; i+1 < length;){
exchange(a,i,--length);
maxHeap(a,i,length);
}
System.out.println(Arrays.toString(a));
}
利用堆构造优先队列
// 把元素x插入到集合A中
public void insert(int[] A,int x){
int i;
// 遍历一遍数组,查找已存在数组的末尾下标
for(i = 0; i < A.length; i++){
if(A[i] == -1){
break;
}
}
A[i] = Integer.MIN_VALUE;
increaseKey(A,i,x);
}
public int maximum(int[] A){
return A[0];
}
extactMax(S) : 去掉并返回S中具有最大关键字的元素,我们结合之前的maxHeap()方法,很容易想到如何实现这个方法。 // 返回最大优先队列中的最大元素并删除
public int extractMax(int[] A){
int max;
int length = A.length-1;
// 判断A数组中元素的个数
if(A.length == 0){
return Integer.MIN_VALUE;
}
max = A[0];
// 交换两个数的值
exchange(A,0,length);
// 重新维护堆的结构
maxHeap(A,0,length);
return max;
}
increaseKey(S,x,k) : 将元素x的关键字值增加到k,这里假设k的值不小于x的原关键字的值。 public void increaseKey(int[] A,int i,int key){
if(key < A[i-1]){
System.out.println("new key is smaller than key");
}
A[i-1] = key;
while(i > 0 && A[i-1] > A[i>>1]){
exchange(A,i-1,i/2);
i = i>>1;
}
}