第4节课仍然是讲排序,但介绍的是一种很高效的堆排序。

在编程过程中,有时候会需要进行extrat_max的操作,即从一个数列里挨个抽取最大值并将其它从原数列中移除。而排序问题也可以看作是一个extract_max的行为,不断的从原始数列中抽取最大值并进行移除,这样挨个抽取的最大值输出后能得到一个降序的数列。为了实现该排序思路,堆排序被提了出来,首先我们的了解下堆的概念:

堆(Heap):一个数列被可视化为一个近似完全二叉树,这个树则为堆。如下图所示:

在堆的基础上,有分:最大堆和最小堆。

  • 最大堆(Max-Heap):某节点的key 子节点的key。
  • 最小堆(Min-Heap):某节点的key  子节点的key。

为了构建最大堆,有个操作叫max_heapify,它就在子树根(subtree's root)上纠正违反最大堆属性(某节点的key ≥ 子节点的key),下图展示了一个max_heapify的例子。

从上图可以看出,倒数第二行(即n/2个元素)是向下比较一次,倒数第三行(n/4个元素)是向下比较两次,...。(问:为什么倒数第三行是向下比较两次?答:用上图的例子说明,如果倒数第三行的4需要修正最大堆属性,和14换,这是比较一次,而换后的4与8不合最大堆属性,所以又换了一次,这是第二次比较。加起来就两次比较)。那么n/2^k个元素的向下比较了k次,所以MAX_HEAPIFY(A, 跟节点)的复杂度Οlog2n(n/2^k≤1)。

对无序数列构建一个大堆堆的方法如下图所示,是以bottom-up的方式,从i=n/2结点位置依次向上遍历进行max_heapify操作

基于以上的方法,我们可以实现堆排序

  1. 将无序A数列构建一个堆。(Ο(n))
  2. 将堆进行max_heapify修正为最大堆。(Ο(log2n))
  3. 找到最大值元素A[1],即当前最大堆的跟节点。(Ο(1))
  4. 将A[1]和堆最后的元素A[n]进行交换。(Ο(1))
  5. 将元素A[n]撤出堆,减少堆大小。(Ο(1))
  6. 现在堆也许会违背最大堆属性,因此重复2-5步骤。(有n steps)

由于A(n)不断撤出,堆大小也会不断改变。复杂度不再是Ο(n + nlog2n) - 其中n为建堆,nlog2n中的第一个n是n steps,而第二个log2n是max_heapify,因为堆大小不断减小时,n和n都在同步减少,这样nlog2n实际上应为n,最后堆排序的时间复杂度为Ο(n)(Ο(n+n)=Ο(2n)=Ο(n))。

下图为一个堆排序的例子:

[MIT6.006] 4. Heaps and Heap Sort 堆,堆排序的更多相关文章

  1. PAT Advanced 1098 Insertion or Heap Sort (25) [heap sort(堆排序)]

    题目 According to Wikipedia: Insertion sort iterates, consuming one input element each repetition, and ...

  2. [MIT6.006] 6. AVL Trees, AVL Sort AVL树,AVL排序

    之前第5节课留了个疑问,是关于"时间t被安排进R"的时间复杂度能不能为Ο(log2n)?"和BST时间复杂度Ο(h)的关系.第6节对此继续了深入的探讨.首先我们知道BST ...

  3. 【算法】堆排序(Heap Sort)(七)

    堆排序(Heap Sort) 堆排序(Heapsort)是指利用堆这种数据结构所设计的一种排序算法.堆积是一个近似完全二叉树的结构,并同时满足堆积的性质:即子结点的键值或索引总是小于(或者大于)它的父 ...

  4. 1098 Insertion or Heap Sort (25 分)(堆)

    这里的第二序列相当于是排序还没拍好的序列 对于第二个样例的第二个序列其实已经是大顶堆了 然后才进行的堆排序 知道这个就好做了 #include<bits/stdc++.h> using n ...

  5. [数据结构]——堆(Heap)、堆排序和TopK

    堆(heap),是一种特殊的数据结构.之所以特殊,因为堆的形象化是一个棵完全二叉树,并且满足任意节点始终不大于(或者不小于)左右子节点(有别于二叉搜索树Binary Search Tree).其中,前 ...

  6. [Unity][Heap sort]用Unity动态演示堆排序的过程(How Heap Sort Works)

    [Unity][Heap sort]用Unity动态演示堆排序的过程 How Heap Sort Works 最近做了一个用Unity3D动态演示堆排序过程的程序. I've made this ap ...

  7. 09-排序3 Insertion or Heap Sort

    和前一题差不多,把归并排序换成了堆排序.要点还是每一次排序进行判断 开始犯了个错误 堆排序该用origin2 结果一直在排序origin ,误导了半天以为是逻辑错误...一直在检查逻辑 建立最大堆 排 ...

  8. 堆排序 Heap Sort

    堆排序虽然叫heap sort,但是和内存上的那个heap并没有实际关系.算法上,堆排序一般使用数组的形式来实现,即binary heap. 我们可以将堆排序所使用的堆int[] heap视为一个完全 ...

  9. 数据结构 - 堆排序(heap sort) 具体解释 及 代码(C++)

    堆排序(heap sort) 具体解释 及 代码(C++) 本文地址: http://blog.csdn.net/caroline_wendy 堆排序包括两个步骤: 第一步: 是建立大顶堆(从大到小排 ...

随机推荐

  1. 迅雷bt种子的制作

    BT是目前最热门的下载方式之一,它的全称为"BitTorrent"简称"BT",中文全称"比特流",但很多朋友将它戏称为"变态下载 ...

  2. MeteoInfoLab脚本示例:Trajectory

    示例读取HYSPLIT模式输出的气团轨迹数据文件,生成轨迹图层,并显示轨迹各节点的气压图.脚本程序: f = addfile_hytraj('D:/MyProgram/Distribution/jav ...

  3. day39 Pyhton 并发编程02 后

    一.开启子进程的另一种方式 import os from multiprocessing import Process class MyProcess(Process): def __init__(s ...

  4. centos8平台redis5日志按天分割

    一,创建日志的备份目录 [root@yjweb crontab]# mkdir /data/logs/redislogsbackup 说明:刘宏缔的架构森林是一个专注架构的博客,地址:https:// ...

  5. 企业内部新建DNS服务器

    DNS软件bind isc 开源 免费使用 其他:powerdns(基于php) undound 安装bind yum list all bind 官方最新版本 www.isc.org/downloa ...

  6. Elasticsearch(5):添加文档

      1 ES数据读写流程¶ ES中,每个索引都将被划分为若干分片,每个分片可以有多个副本.这些副本共同组成复制组,复制组中的分片在添加或删除文档时必须保持同步,否则,从一个副本中读取的数据将与从另一个 ...

  7. 假如 Web 当初不支持动态化

    楔子 Web 生而具有极其灵活的动态化基础能力,诸如: 动态插入script标签执行任意脚本逻辑 动态插入style标签引入任何 CSS 样式规则 通过iframe标签嵌入整站 以上标签均可直接加载网 ...

  8. CRC(循环冗余校验)

    关于CRC(循环冗余校验),我在网上看了许多的文章,感觉看的很懵逼,废话一堆(可能是我理解不上去0.0),下面是我的一些理解(如果有误谢谢指出): 关于crc,它主要分为两个部分,一个是发送端通过cr ...

  9. ceil中有-0啊

          这里主要是有一点: 1 Math.ceil(d1)  ceil 方法上有这么一段注释:If the argument value is less than zero but greater ...

  10. CF1430 D. String Deletion(div 2)

    题目链接:http://codeforces.com/contest/1430/problem/D 题意:有一个长度为n(n<=2*10^5)的01字符串,每轮操作有两步: 第一步是删去字符串中 ...