堆排序
堆性质:
1.是一个顺序存储的完全二叉树
2.每个节点都大于或等于它的任意子节点是大顶堆,反之是小顶堆。
算法思想(以大顶堆为例):
1.首先是一个无序的数组,画成完全二叉树的形式
由上图可以看出:
设某节点下表为i,则其左孩子下表为2i+1,右孩子下表为2i+2。
2.构建大顶堆:调整堆使每个节点大于或等于它的任意孩子
这里我采用的是从下往上调整,经过一次循环可以把最大值调整到堆顶。
经过一次遍历循环,从下往上调整那么必然可以选出一个最大值放在堆顶,然后去掉第一个把后面的数据再进行递归直到只剩一个节点就结束递归。
void Swap(int* x,int *y)//交换数据
{
int temp = *x;
*x = *y;
*y = temp;
}
void HeapCreate(int *arr, int size)
{
if(size == 1)
return;
int i;
int n = size%2==0 ? (size-1)/2 : (size-2)/2;//找到最后一个有孩子的节点
i = n;
while(i >= 0){
if(arr[i] < arr[2*i+1]){ //与左孩子比较
Swap(&arr[i],&arr[2*i+1]);
}
//由于循环开始是最后一个有孩子的节点,所以这个节点必有左孩子,但是不一定有右孩子,所以还要判断一下防止数组越界访问
if(arr[i] < arr[2*i+2] && 2*i+2 < size){//与右孩子比较,
Swap(&arr[i],&arr[2*i+2]);
}
i--;
}
HeapCreate(arr+1, size-1);//找出了最大值,把剩下的节点递归
//其实就是第一遍从下标0开始选出最大值到arr[0],然后再从下标1开始选出次大值到arr[1]......
}
3.个人感受:堆排序和选择排序很类似,只是堆排序在每次调整中已经相对有序了。所以比选择排序少了一些频繁的交换。