算法分析之——heap-sort堆排序
堆排序是一种原地排序算法,不使用额外的数组空间,运行时间为O(nlgn)。本篇文章我们来介绍一下堆排序的实现过程。
要了解堆排序。我们首先来了解一个概念,全然二叉树。
堆是一种全然二叉树或者近似全然二叉树。
什么是全然二叉树呢?百度百科上给出定义:全然二叉树:除最后一层外,每一层上的节点数均达到最大值;在最后一层上仅仅缺少右边的若干结点。以下用两个小图来说明全然二叉树与非全然二叉树。
(图片来自百度,大家能够忽略水印…..)
二叉堆满足二个特性:
1.父结点的键值总是大于或等于(小于或等于)不论什么一个子节点的键值。
2.每一个结点的左子树和右子树都是一个二叉堆(都是最大堆或最小堆)。
当父结点的键值总是大于或等于不论什么一个子节点的键值时为最大堆。
当父结点的键值总是小于或等于不论什么一个子节点的键值时为最小堆。
堆排序也是基于分治思想的。主要有以下三步:
- 初始化,从第一个非叶结点開始遍历,使以其为根的树为大根堆;
- 交换堆顶元素与堆尾元素。筛选出最大的值。调整新的堆为大根堆。
- 反复2,每次筛选出堆中的最大元素,堆排序完毕。
本次。我们举例的数组例如以下:数组长度length=10。
相应的堆结构:
从第一个非叶结点開始。建初堆
void BUILD_MAX_HEAP(int A[],int length)
{
int i;
for(i=((length/2)-1);i>=0;i--)//length/2-1,为第一个非叶结点
MAX_HEAPIFY(A,i);
}
保持堆的性质
void MAX_HEAPIFY(int A[],int i)
{
int l,r,largest,middle;
l=LEFT(i);
r=RIGHT(i);
if(l<heap_size && A[l]>A[i])
largest = l;
else
largest= i;
if(r<heap_size && A[r]>A[largest])
largest = r;
if(largest!=i)
{
middle=A[largest];
A[largest]=A[i];
A[i]=middle;
MAX_HEAPIFY(A,largest);
}
}
堆排序的详细实现
void heap_sort(int A[],int length)
{
BUILD_MAX_HEAP(A,length);
int i,middle;
for(i=length-1;i>0;i--)
{
middle=A[0];
A[0]=A[i];
A[i]=middle;
heap_size--;
MAX_HEAPIFY(A,0);
}
}
以下为程序运行的简单的过程。分析不够全面,可是足以说明问题。
1.分析步骤4中的for循环,BUILD_MAX_HEAP(A,length);即建初堆的过程。i从length/2-1循环到0,即从4循环到0。(4为第一个非叶结点)
(1)i=4。MAX_HEAPIFY(A,i)。MAX_HEAPIFY(A,4);
<1>计算左叶子节点的编号l=LEFT(i)=(2*i+1)=9; 计算右叶子节点的编号r=RIGHT(i)=(2*i+2)=10;
注:此处计算左右叶子节点的编号时,要注意数组是从0还是从1開始的。若从0開始。左叶子节点为(2*i+1),右叶子节点为(2*i+2)。若从1開始,左叶子为2*i;右叶子为2*i+1
<2>推断左右叶子节点与根节点的大小,将当中节点编号的较大值赋值给largest;
heap_size为堆的大小,開始heap_size=length=9
if(l<heap_size && A[l]>A[i])
largest = l;
else
largest= i;
if(r<heap_size && A[r]>A[largest])
largest = r;
当i=4时,largest=4
<3>推断largest是否等于根节点,若不为根节点。说明当中左叶节点或者右叶节点比根节点的值大,则此时交换根节点与largest节点的值。
if(largest!=i)
{
middle=A[largest];
A[largest]=A[i];
A[i]=middle;
MAX_HEAPIFY(A,largest);
}
由于此处largest=i,因此不运行这一步,运行下一次for循环
(2)i=3;MAX_HEAPIFY(A,i);MAX_HEAPIFY(A,3);
<1>计算左叶子节点的编号l=LEFT(i)=(2*i+1)=7; 计算右叶子节点的编号r=RIGHT(i)=(2*i+2)=8;
<2>推断左右叶子节点与根节点的大小。将当中节点编号的较大值赋值给largest;
largest=7
<3>推断largest是否等于根节点,若不为根节点。说明当中左叶节点或者右叶节点比根节点的值大,则此时交换根节点与largest节点的值。
<4>运行MAX_HEAPIFY(A,largest);MAX_HEAPIFY(A,7);将以largest为根的树调整为大根堆
(3)i=2;步骤与2中的<1>~<4>同样。largest=6,发生交换。
此处不再分析。
(4)i=1;时分析过程參考步骤与2中的<1>~<4>。
运行后的结果
(5)i=0;
2.步骤6中的for循环分析。即筛选出最大的值。缩小堆的规模,保持堆的性质的过程。
length=10,i从length-1到1。即从9循环到1
for(i=length-1;i>0;i--)
{
middle=A[0];
A[0]=A[i];
A[i]=middle;
heap_size--;
MAX_HEAPIFY(A,0);
}
(1)i=9;交换A[i]与A[0]。此时i是堆的最末的那个元素,A[0]是堆顶元素,即最大的元素,将最大元素交换到堆尾。而且堆的规模缩小一个,即此时待又一次排序的堆是红框框起来的部分,此时运行MAX_HEAPIFY(A,0);上面已经分析,此处不再赘述。
(2)i=8;交换A[i]与A[0];heap_size–;MAX_HEAPIFY(A,0);
(3)以下的循环不再举例,我们能够看出。每次都筛选出当前堆中最大的元素。
3.最后给出程序运行的截图:
程序源码下载地址:堆排序实现代码
算法分析之——heap-sort堆排序的更多相关文章
- [Unity][Heap sort]用Unity动态演示堆排序的过程(How Heap Sort Works)
[Unity][Heap sort]用Unity动态演示堆排序的过程 How Heap Sort Works 最近做了一个用Unity3D动态演示堆排序过程的程序. I've made this ap ...
- 算法 Heap sort
// ------------------------------------------------------------------------------------------------- ...
- 堆排序 Heap Sort
堆排序虽然叫heap sort,但是和内存上的那个heap并没有实际关系.算法上,堆排序一般使用数组的形式来实现,即binary heap. 我们可以将堆排序所使用的堆int[] heap视为一个完全 ...
- 数据结构 - 堆排序(heap sort) 具体解释 及 代码(C++)
堆排序(heap sort) 具体解释 及 代码(C++) 本文地址: http://blog.csdn.net/caroline_wendy 堆排序包括两个步骤: 第一步: 是建立大顶堆(从大到小排 ...
- 堆排序(Heap Sort)的C语言实现
堆排序(Heap Sort)具体步骤为 将无序序列建成大顶堆(小顶堆):从最后一个非叶子节点开始通过堆调整HeapAdjust()变成小顶堆或大顶堆 将顶部元素与堆尾数组交换,此是末尾元素就是最大值, ...
- PAT A1098 Insertion or Heap Sort (25 分)——堆排序和插入排序,未完待续。。
According to Wikipedia: Insertion sort iterates, consuming one input element each repetition, and gr ...
- PAT甲题题解1098. Insertion or Heap Sort (25)-(插入排序和堆排序)
题目就是给两个序列,第一个是排序前的,第二个是排序中的,判断它是采用插入排序还是堆排序,并且输出下一次操作后的序列. 插入排序的特点就是,前面是从小到大排列的,后面就与原序列相同. 堆排序的特点就是, ...
- 算法----堆排序(heap sort)
堆排序是利用堆进行排序的高效算法,其能实现O(NlogN)的排序时间复杂度,详细算法分析能够点击堆排序算法时间复杂度分析. 算法实现: 调整堆: void sort::sink(int* a, con ...
- PAT甲级——1098 Insertion or Heap Sort (插入排序、堆排序)
本文同步发布在CSDN:https://blog.csdn.net/weixin_44385565/article/details/90941941 1098 Insertion or Heap So ...
- Python入门篇-数据结构堆排序Heap Sort
Python入门篇-数据结构堆排序Heap Sort 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.堆Heap 堆是一个完全二叉树 每个非叶子结点都要大于或者等于其左右孩子结点 ...
随机推荐
- PHP中的命名空间(namespace)及其使用详解
PHP中的命名空间(namespace)及其使用详解 晶晶 2年前 (2014-01-02) 8495次浏览 PHP php自5.3.0开始,引入了一个namespace关键字以及__NAMESPAC ...
- 【Objective-C】03-第一个OC程序
一.打开Xcode,新建Xcode项目 二.选择最简单的命令行项目 因为我们只是学习OC语法,还未正式进入iOS开发,所以选择命令行项目即可 三.输入项目名称,选择Foundation框架进行创建项目 ...
- 解决:eclipse配置Database Connections报错Communications link failure Last packet sent to the server was 0 ms ago
网上各式各样的问题,不过我的问题在于我开了Proxifier,导致链接localhost的时候被拦截...把Proxifier关了就好了 以后遇到这种问题.连不上数据库啊,连不上本地的服务器啊,先检查 ...
- springboot2.0 fastjson 日期格式化不生效解决
网上有很多介绍,主要有两种. 1.在启动类中注入bean /** * 集成fastjson * * @return */ @Bean public HttpMessageConverters fast ...
- CentOS平滑升级Nginx
服务器:CentOS 6.4 64位 升级方案:nginx1.4.0 – nginx1.4.3 Nginx编译后就一个小文件,不带动态库,升级也可以无缝升级,并不影响访问,按下面的命令执行就可以,具体 ...
- 使用python进行新浪微博粉丝爬虫
由于最近没事在学python,正好最近也想趴下新浪微博上边的一些数据,在这里主要爬去的是一个人的粉丝具体信息(微博昵称,个人介绍,地址,通过什么方式进行关注),所以就学以致用,通过python来爬去微 ...
- datatable.js 服务端分页+fixColumns列固定
前言 记得还是15年的时候,工作需要,独自写后台管理系统..记得那时候,最让我头疼的不是后台逻辑,而是数据的列表展示. 列很多的时候,页面显示问题;分页样式问题;表格样式问题;数据加载...很多细节的 ...
- textarea去掉边框
<textarea style="BORDER-BOTTOM: 0px solid; BORDER-LEFT: 0px solid; BORDER-RIGHT: 0px solid; ...
- 支付宝API接口开发相关文档以及实例
支付宝实物即时到账接口,在下载该接口之前,请确定你已经签约了即时到账的服务协议,并且账号已经开通了即时到账的权限. 口常见错误问题搜集:http://union.alipay.com/alipay/z ...
- 【转】MYSQL 存储过程定时操作数据库
这个涉及2个步骤,第一个就是建立存储过程: create procedure clear_table() begin drop database XXX end 第二步就是让其定时运行: 查看even ...