堆排序算法C语言版本
前置知识点:完全二叉树
在完全二叉树中,结点i的左孩子为2i,右孩子为2i+1,双亲为i/2取整。
#include <stdio.h>
void heapsort(int num[],int len);
void heapAdjust(int num[],int s,int n);
int main(){
/*因为完全二叉数层序遍历第一个元素下标为1
,所以传入数组的0号元素不参与排序*/
int num[11] = {-1,5,2,6,0,3,9,1,7,4,8};
heapsort(num,10);//传入数组和需排序元素个数
for(int i = 1; i<11 ;i++){//输出排序后1-10号元素
printf("%d ",num[i]);
}
}
void heapsort(int num[],int n){
/*对数组num进行堆排序,n为数组需排序元素个数
,初始n=length-1,从小到大排序,O(nlog2^n),不稳定*/
/*核心思想,因为完全二叉数层序遍历第一个元素下标为1
,数组的0号元素不参与排序,完全二叉树双亲下标为s
,左孩子为s*2,右孩子为s*2+1,heapAdjust()只能比较双亲和孩子
,循环调用从下往上、从后往前构造出大顶堆,构造出大顶堆后
,第1号元素也就是未排序区段最大值,将其放至最后并且不再参与排序
,然后继续循环调用heapAdjust并交换值,就能使数组前小后大
*/
int i,temp;
for(i=n/2; i>0; i--){
//从下往上构造大顶堆,初始i为最后一个元素的双亲
heapAdjust(num,i,n);//传入数组,双亲下标,需排序元素个数
}
for(i = n; i>1; i--){
//此时1号元素最大,将其放到最后
temp = num[1];
num[1] = num[i];
num[i] = temp;
heapAdjust(num,1,i-1);//再次构造大顶堆,最后一个元素不参与
}
}
void heapAdjust(int num[],int s,int n){
/*构造大顶堆方法,s是双亲下标,n是需排序元素个数
,找出s的子孙最大值交换至s位置
*/
int i,temp;//i用于存放比双亲大的孩子的下标
temp = num[s];//temp存放双亲数值
for(i=2*s; i<n; i*=2){//i每次都为双亲的左孩子,i<n不能越界
if(num[i] < num[i+1]){//比较左右孩子大小
i++;//右孩子大于左孩子,i存放右孩子下标
}
if(temp >= num[i]){//如果双亲大于等于左右孩子
break;//不需要变动
}
num[s] = num[i];//将比双亲大的孩子数值存入双亲结点
s = i;//继续往完全二叉树下层循环
}
num[s] = temp;//全部循环完后,将初始双亲数值存入相应位置
}