《算法4》中的堆排序

原创 2015年07月10日 11:04:42

因为要写STL中的heap,所以又将这本书中的归并排序复习了一下,并且有所收获,为了避免忘记,我准备将这些思路记下来:
堆排序有两个重要的动作:swimsink
下面写一下两个函数

void swim(int* num, int k, int N)
{
    while(k > 1 && a[k/2] < a[k])
    {
        swap(a[k/2], a[k]);
        k = k/2;
    }
}
void sink(int* num, int k, int N)
{
    while(2*k <= N)
    {
        int j = 2*k;
        if(2*k < N && num[2*k] < num[2*k+1]) 
            j++;
        if(num[k] >= num[j])
            break;
        swap(num[k], num[j]);
        k = j;
    }
}

有了这两个函数,我们就可以构建堆排序了;
首先要构建堆,在构建堆的过程中,数组从左向右遍历并利用swim函数时可以构造成一个堆的,但这需要遍历数组中的全部数字;然而采用数组从右向左遍历并利用sink函数就可以仅仅遍历一般的数字,因为叶子节点不需要进行sink
对于构建堆需要的复杂度是O(2N),因为需要2N次的比较和N次的交换(交换和比较时一个算法中的主要成本,如果没有这两者再考虑存取的成本),这里可以举例:对于一个127个元素的堆,在构建的时候:32个大小为3的堆(需要一次交换,两次比较),16个大小为7的堆(两次交换三次比较),8个大小为15的堆,4个大小为31的堆,2个大小为63的堆和1个大小为127的堆,一共就需要120交换和240比较。

之后就可以开始排序了:首先将堆顶的元素和数组的尾部元素交换,然后刷新数组的长度,让算法无法再访问这个数组尾部元素;
将每次都从最高的点插入堆尾的元素,然后利用sink下沉,重新adjust这个堆,以此类推,可以参考我这段代码;

while(N > 1)
{
    swap(num[1], num[N--]);
    sink(num, 1, N);
}

总体的思路就是这个样子,但是,如果遇到“比较”的成本特别大的时候,还可以采用如下的方法:

先下沉后上浮

我们可以这样来思考,处在数组尾部的元素一般都是比较小的元素,这样我们在把它放到堆顶后进行sink,一般它还是会下来的,那么我们为什么不直接将这个元素放到堆尾然后在swim上去呢?这样会减少很多次的比较,具体做法如下:
1.当我们将堆顶的元素放在数组尾部后,我们利用辅助空间暂时存储之前的堆尾元素temp,并不是将它放在堆顶;
2.现在堆顶是空的,我们不将temp放在上面,而是直接将堆顶对应的子节点中较大的那个节点bigger直接放上去(这样我们只需要比较一次哪个子节点大就可以了),然后这个bigger的位置就产生了空缺;
3.这个空缺的左右子节点继续比较产生较大的节点bigger`放在父节点空缺的位置上;
4.依次类推直到空缺的节点没有子节点;
5.将空缺的节点放上temp;
6.swim上浮
这个方法需要一个辅助空间,如果条件允许的话便可以用;而这种方法产生的比较次数几乎就成为了N;这个方法是Floyd在1964年改善的,我觉得十分不错;只是《算法4》中的描述我不太习惯,换成我习惯方式^^;

相关文章推荐

【算法总结】堆及堆排序总结

【前言】 堆排序是什么? 堆排序的核心是将数组构建成为一个堆,然后从堆顶逐个逐个数字获取,堆分成最大堆(大顶堆)及最小堆(小顶堆)。 下面还是先说明什么是堆。 堆是...
  • abcd1f2
  • abcd1f2
  • 2015年08月03日 18:18
  • 1021

堆排序 两种实现(最小堆和最大堆)

堆排序算法是建立在二叉树的堆结构上的,通过交换堆(一维数组)中的元素,并进行上浮或下沉函数运算实现多次调整 堆排序算法的复杂度低,和快速排序属于相同速度级别的一种快速的排序算法 下面以最小堆和最大...

读书笔记_算法第四版(一)

算法第四版(谢路云译) 官方网站:http://algs4.cs.princeton.edu/home/有部分源代码和部分课后习题答案。 个人练习代码:https://github.com/mor...

八、优先队列、堆排序

优先队列 API 初级实现 堆得定义 堆的算法 索引优先队列 索引优先队列的使用案例 堆排序优先队列一种常见的数据结构,需要支持两种操作:删除最大(最小)元素和插入元素。这种数据类型叫做优先队列。AP...

算法(第四版)学习笔记之java实现堆排序

继上一篇实现基于堆的优先队列后,这次将利用上一次完成的基于堆的能够重复删除最大元素操作的优先队列来实现一种经典而优雅的排序算法,称之为堆排序。 堆排序可分为两个阶段: 1.构建堆:在堆的构建过程中...

八大排序算法(4) 堆排序

堆:父节点 不大于/不小于 其子节点的完全二叉树。 利用堆的特性,可快速选择出序列中的最大最小元素。...

数据结构算法之排序系列Java、C源码实现(4)--堆排序

堆是一种树形结构,堆排序是对直接选择排序的有效改进。实现堆排序需要解决三个问题: 1.构建完全二叉树:先把待排序序列构建成一棵完全二叉树。    2.把无序序列建成的完全二叉树调成一个有序堆。 然后根...

java版排序算法简介及冒泡排序以及优化,选择排序,直接插入排序,希尔排序,堆排序,快速排序及其优化前言 2 分类 2 稳定性 3 时间复杂度 4 Java实现版本 5 1、冒泡排序 6 2、选择排序

写在前边滴~~学习小甲鱼视频笔记,链接:http://bbs.fishc.com/portal.php 然后图文并茂,这是我自己的笔记so~~发表出来是为了更快的找到我自己的总结,所以想看官方解释的,...

C语言8种排序算法及其实现 1.希尔排序 2.二分插入法 3.直接插入法 4.带哨兵的直接排序法 5.冒泡排序 6.选择排序 7.快速排序 8.堆排序

一.希尔(Shell)排序法(又称宿小增量排序,是1959年由D.L.Shell提出来的) /* Shell 排序法 */ #include void sort(int v[],int n) { ...

堆排序算法c语言实现

  • 2015年10月25日 14:24
  • 2KB
  • 下载
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:《算法4》中的堆排序
举报原因:
原因补充:

(最多只允许输入30个字)