堆排序算法c语言筛选法,【排序】排序算法之选择排序

排序算法之选择排序

罗朝辉(http://www.cppblog.com/kesalin)

转载请注明出处

排序是数据处理中经常使用的一种重要运算,在计算机及其应用系统中,花费在排序上的时间在系统运行时间中占有很大比重,其重要性无需多言。下文将介绍常用的如下排序方法,对它们进行简单的分析和比较,并提供 C 语言实现。

所谓排序,就是要将一堆记录,使之按关键字递增(或递减)次序排列起来。根据排序所采用的策略,可以分为如下五种:

3、选择排序(直接选择排序、堆排序)

4、归并排序;

5、桶排序(桶排序,基数排序);

---------------------------------------------------------------------------------

前面讲了插入,交换排序,下面接着来讲选择排序。

选择排序(Selection Sort)的基本思想是:每一趟从待排序的记录中选出关键字最小的记录,顺序放在已排好序的子文件的最后,直到全部记录排序完毕。

常用的选择排序方法有直接选择排序和堆排序。

直接选择排序

基本思想:将待排序记录分成有序区和无序区,初始状态下有序区位空,无序区为整个待排序记录。每一趟选择排序就是在无序区中选择最小(或最大)的记录,插入到有序区的最后。如此循环直到无序区为空,完成排序。

代码实现

//直接选择排序//voidselect_sort(int*array,intlength)

{

assert(array&&length>=0);inti, j, k, temp;for(i=1; i

k=i;for(j=i+1; j

k=j;

}

}if(k!=i) {

temp=array[i];

array[i]=array[k];

array[k]=temp;

}

}

}

时间复杂度分析:

无论待排序记录的初始状态如何,在第 i 趟排序中选出最小关键字的记录,需做 n - i 次比较,因此,总的比较次数为: n * (n - 1) / 2 = 0(n ^ 2),当待排序记录的初始状态为正序时,移动次数为 0;当初始状态为反序时,每趟排序均要执行交换操作,总的移动次数取最大值 3 * (n-1)。所以,直接选择排序的平均时间复杂度为 0(n ^ 2)。

空间复杂度:

很明显,0(1)。

补充:

直接选择排序是一个就地排序,且是非稳定排序。

堆排序

堆的定义:满足如下约束的 n 个关键字序列 Kl,K2,…,Kn称为堆,1 ≤ i ≤n/2,

(1) ki≤ K2i且 ki≤ K2i+1 (小根堆)   或

(2) Ki≥ K2i且 ki≥ K2i+1 (大根堆)

从定义来看,堆实质上是满足如下性质的完全二叉树:树中任一非叶结点的关键字均不大于(或不小于)其左右孩子(若存在的话)结点的关键字。堆排序就是充分利用堆这种堆顶记录要么是最大的要么是最小的有序性质,每次在堆中选择最大或最小关键字完成排序。堆排序也是选择排序的一种,它比直接选择排序平均效率要高,因为直接选择排序,每次比较之后并没有对比较结果进行保存,导致可能存在重复的比较,而堆则利用其堆性质将比较结果保存在堆中了(非叶子节点的左边孩子一定不大于或不小于比右边的孩子)。

堆排序的关键就在于将待排序记录的建立成堆,建好堆之后,将无序区的堆顶记录(总是无序区的最大或最小)和无序区最后一个记录交换,交换之后,无序区的最后一个记录变成有序区的第一个记录,而无序区新的的堆顶记录不一定满足堆性质了,因而需要将无序区调整而堆。如此循环,直到无序区为空,结束排序。

所以堆排序的主要步骤就分三步:

1,建立初始堆;

2,将无序的堆顶记录与最后一个记录交换,缩小无序区;

3,将交换之后的无序区调整为堆,重复步骤 2。

代码实现:

//筛选法调整堆,除 [low] 之外,[low] 的两个孩子均已是大根堆voidadjust_heap(int*heap,intlow,inthigh)

{

assert(heap);#if1//循环实现inti=low;intj=2*i;inttemp=heap[i];while(j<=high) {//若有两个孩子,j 为孩子中大的那个的下标if(j

j=j+1;

}//已是堆if(temp>=heap[j]) {break;

}//继续筛选else{

heap[i]=heap[j];

i=j;

j=2*i;

}

}

heap[i]=temp;#else//递归实现inti=low;intj=2*i;inttemp=heap[i];if(j>=high) {return;

}//若有两个孩子,j 为孩子中大的那个的下标if(jheap[j]) {

j=j+1;

}//已经为堆,无需调整if(heap[low]>=heap[j]) {return;

}

heap[i]=heap[j];

heap[j]=temp;//调整之后,[j, high] 可能不满足堆了,需继续调整adjust_heap(heap, j, high);#endif}//只有一个结点的树是堆,而在完全二叉树中,所有序号 i > n/2 的结点都是叶子,//因此以这些结点为根的子树均已是堆。这样,我们只需依次将以序号为//n/2, n/2 - 1, …, 0 的结点作为根的子树都调整为堆即可。voidbuild_heap(int*heap,intlength)

{

assert(heap&&length>=0);inti;for(i=length/2; i>=0;--i) {

adjust_heap(heap, i, length-1);

}

}//堆排序//voidheap_sort(int*array,intlength)

{

assert(array&&length>=0);if(length<=1) {return;

}inti, temp;//将 [0, length - 1] 建成初始堆build_heap(array, length);//对当前无序区 [0, i - 1] 进行堆排序,共做 length - 1 趟。for(i=length-1; i>0;--i) {//将堆顶和堆中最后一个记录交换temp=array[0];

array[0]=array[i];

array[i]=temp;//将 [0, i - 1] 重新调整为堆,仅有 [0] 可能违反堆性质adjust_heap(array,0, i-1);

}

}

时间复杂度分析:

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值