数据结构——堆与堆排序

最近开始学习王争老师的《数据结构与算法之美》,通过总结再加上自己的思考的形式记录这门课程,文章主要作为学习历程的记录。

堆是一种特殊的树,只要满足以下两点,它就是一个堆:

一、堆是一个完全二叉树

二、堆中每一个节点的值都必须大于等于(或小于等于)其子树中每个节点的值。

对于每个节点的值都大于等于子树中每个节点值的堆,叫作“大顶堆”。

对于每个节点的值都小于等于子树中每个节点值的堆,叫作“小顶堆”。

如何实现一个堆?

要实现一个堆,要先知道堆都支持哪些操作以及如何存储一个堆。

完全二叉树通常用数组来存储,如下图:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-LVo369Wg-1574929603023)(E:\数据结构(python)\images\堆1.jpg)]

数组中下标为i的节点的左子节点,就是下标为 i ∗ 2 i*2 i2的节点,右子节点就是 i ∗ 2 + 1 i*2+1 i2+1的节点。父节点就是下标为 i / 2 i/2 i/2的节点。

1.往堆中插入元素

往堆中插入一个元素后,我们需要继续满足堆的两个特性。把新插入的元素放在堆的最后,但这样不符合堆的特性,需要进行堆化。堆化实际上分为两种:从下往上从上往下。堆化就是顺着节点所在路径,向上或向下,对比然后交换。

先讨论从下往上的堆化方法:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-C9vJi9pZ-1574929603024)(E:\数据结构(python)\images\堆2.jpg)]

让新插入的节点与父节点对比大小。如果不满足子节点小于等于父节点的大小关系,就互换两个节点。一直重复这个过程,直到父子节点之间满足堆的大小关系。

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-qgncGCfd-1574929603026)(E:\数据结构(python)\images\堆3.jpg)]

def insert(data):
    data_list.append(data)   #data_list为数组,从下标1开始存储数据
    index = len(data_list)
    while(index/2>0 and data_list[index]>data_list[index/2]):  #自下往上堆化
        swap(data_list,index,index/2)
        index = index/2

2.删除堆顶元素

把最后一个节点放到堆顶,然后利用同样的父子节点对比方法。

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-bB75f68A-1574929603026)(E:\数据结构(python)\images\堆4.jpg)]

def removeMax():
    if count==0: #count表示堆中已经存储的数据个数
        return -1
    a[1] = a[count]
    count-=1
    heapify(data_list,count,1)
    
def heapify(a,n,i):
    while True:
        maxvalue_index = i
        if i*2 <= n and a[i] < a[i*2]:
            maxvalue_index = i*2
        if i*2+1 <= n and a[maxPos] < a[i*2+1]:
            maxvalue_index = i*2+1
        if maxvalue_index==i:
            break
        swap(a,maxvalue,i)
        i = maxvalue_index

如何基于堆实现排序?

借助于堆这种数据结构实现的排序算法,就叫做堆排序。堆排序的过程分解为两个大步骤:建堆排序

一、建堆

首先将数组原地建成一个堆。建堆的过程有两种思路:

第一种是借助在堆中插入元素的思路,尽管数组中包含n个数据,但可以假设堆中只包含一个数据,就是下标为1的数据,然后调用前面讲的插入操作,将下标从2到n的数据依次插入到堆中,这样就包含n个数据的数组,组织成了堆。

第二种实现思路,是从后往前处理数组,并且每个数据都是从上往下堆化。

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-nJay2tLz-1574929603026)(E:\数据结构(python)\images\堆5.jpg)]
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-XZD6SY6x-1574929603026)(E:\数据结构(python)\images\堆6.jpg)]

def buildHeap(a,n):
    for i in range(int(n/2),0,-1):
        heapify(a,n,i)

代码是对下标从 n / 2 n/2 n/2开始到1的数据进行堆化,下标为 n / 2 + 1 n/2+1 n/2+1到n的节点是叶子节点,不需要堆化。对于完全二叉树,下标从 n / 2 + 1 n/2+1 n/2+1到n的节点都是叶子节点。

因为叶子节点不需要堆化,所以需要堆化的节点从倒数第二层开始。每个节点堆化的过程中,需要比较和交换的个数,跟这个节点的高度k成正比。

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-X9s1zrRx-1574929603026)(E:\数据结构(python)\images\堆7.jpg)]

将每个非叶子节点的高度求和,就是下面这个式子:

S 1 = 1 ∗ h + 2 1 ∗ ( h − 1 ) + 2 2 ∗ ( h − 2 ) + . . . + 2 k ∗ ( h − k ) + . . . + 2 h − 1 ∗ 1 S1 = 1*h + 2^1*(h-1) + 2^2*(h-2) +...+ 2^k*(h-k) + ... + 2^{h-1}*1 S1=1h+21(h1)+22(h2)+...+2k(hk)+...+2h11

通过 2 S 1 − S 1 2S1-S1 2S1S1,得到 S = − h + 2 + 2 2 + . . . + 2 h − 1 + 2 h = 2 h + 1 − h − 2 S = -h + 2 + 2^2 + ... + 2^{h-1} + 2^{h} = 2^{h+1}-h-2 S=h+2+22+...+2h1+2h=2h+1h2

h = l o g 2 n h = log_{2}n h=log2n代入公式S,就得到S=O(n),故时间复杂度为O(n)。

二、排序

建堆结束后,数据中的数据已经是按照大顶堆的特性来组织的。数组中的第一个元素就是堆顶,也就是大元素,把它同最后一个元素交换,那最大元素就放到下标为n的位置。

有点类似于“删除堆顶元素”的操作,当堆顶元素移除后,我们把下标为n的元素放到堆顶,然后再通过堆顶的方法,将剩下的n-1个元素重新构建成堆。堆化完成后,我们再将堆顶的元素放到下标是n-1的位置,一直重复这个过程,直至最后堆中只剩下标为1的一个元素,排序工作就完成了。

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-cyiD5W3l-1574929603027)(E:\数据结构(python)\images\堆8.jpg)]

def sort(a,n):
    buildHeap(a,n)
    k = n
    while(k>1):
        swap(a,1,k)
        k -= 1
        heapify(a,k,1)

每次重建意味着一个节点出堆,堆容量减一。每次重建对调次数与高度k成正比。重建堆需要n-1次循环,相加时间复杂度为 l o g 2 + l o g 3 + . . . + l o g n = l o g ( n ! ) = n l o g n log2+log3+...+logn=log(n!)=nlogn log2+log3+...+logn=log(n!)=nlogn,故排序时间复杂度为O( n l o g n nlogn nlogn)。

堆排序是原地排序,包括建堆和排序。建堆的时间复杂度为O(n),排序的时间复杂度为O( n l o g n nlogn nlogn),故堆排序的时间复杂度为O( n l o g n nlogn nlogn)。

堆排序不是稳定的排序算法,因为在排序过程中,存在将堆的最后一个节点跟堆顶节点互换的操作,因此有可能改变值相同数据的原始相对顺序。

在实际开发中,为什么快速排序要比堆排序性能好?

①堆排序数据访问的方式没有快速排序友好。

②对于同样的数据,在排序过程中,堆排序算法的数据交换次数要多于快速排序。

的最后一个节点跟堆顶节点互换的操作,因此有可能改变值相同数据的原始相对顺序。

在实际开发中,为什么快速排序要比堆排序性能好?

①堆排序数据访问的方式没有快速排序友好。

②对于同样的数据,在排序过程中,堆排序算法的数据交换次数要多于快速排序。

堆的应用

堆这种数据结构有几个非常重要的应用:优先级队列、求Top K和中位数。

堆的应用一:优先级队列

在优先级队列中,数据的出队顺序不是先进先出,而是按照优先级,优先级最高的最先出队。

可以用堆来实现优先级队列。一个堆可以看作一个优先级队列。往优先级队列中插入一个元素,就相当于往堆中插入一个元素;从优先级队列中取出优先级最高的元素,就相当于取出堆顶元素。

1、合并有序小文件

假设有100个小文件,每个文件为100M,每个文件中存储的都是有序的字符串。希望将这100个小文件合并成一个有序的大文件,这就需要优先级队列。

通常我们会从这100个文件中,各取第一个字符串,放入数组中,然后比较大小,把最小的那个字符串放入合并后的大文件,并从数组中删除。如果采用数组这种数据结构,需要循环遍历整个数组,不够高效。

这里就可以用到优先级队列,也可以说是堆。从小文件中取出的字符串放入到小顶堆中,堆顶元素,也就是优先级队列队首的元素,就是最小的字符串。将这个字符串放入到大文件中,并将其从堆中删除,然后再从小文件中取出下一个字符串放入到堆中,循环这个过程,就可以将100个小文件中的数据依次放入到大文件中。删除堆顶数据和往堆中插入数据的时间复杂度是O( l o g n logn logn),n为100,如此操作更为高效。

2、高性能定时器

假设一个定时器,维护了很多定时任务,每个任务都设定了一个要触发执行的时间点。定时器每过一个很小的单位时间(比如说1秒),就扫描一遍任务,看是否有任务到达设定的执行时间。如果到达了,就拿出来执行。

每过1秒扫描一次任务列表的做法比较低效,主要因为:

第一,任务的约定执行可能还要很久。

第二,每次都要扫描整个任务列表,如果任务列表很大,势必会很耗时。

针对这些问题,可以用优先级队列来解决。按照任务设定的执行时间,将这些任务存储在优先级队列中,队列首部(小顶堆堆顶)存储的是最先执行的任务。

拿队首任务的执行时间点与当前时间点相减,得到一个时间间隔T。这样定时器在设定T秒后,再来执行任务。执行完队首的任务后,再计算新的队首任务的执行时间点与当前时间点的差值。如此反复,性能就提高了。

堆的应用二:利用堆求Top K

将求Top K的问题抽象成两类:一类是针对静态数据集合,即数据集合事先确定,另一类是针对动态数据集合,即有数据动态地加入到集合中。

针对静态数据,可以维护一个大小为K的小顶堆,顺序遍历数组,从数组中取出数据与堆顶元素。如果比堆顶元素大,就把堆顶元素删除,并将这个元素插入到堆中。如果比堆顶元素大,则不作处理,继续遍历。等数据都遍历完后,堆中的数据就是前K大数据。遍历数组需O(n)的时间复杂度,一次堆化操作需O( l o g K logK logK)的时间复杂度。最坏的情况下,n个元素都入堆一次,时间复杂度为O( n l o g K nlogK nlogK)。

针对动态数据,求得Top K就是实时Top K。实际上,我们都一直在维护一个K大小的小顶堆。当有数据被添加到集合中,我们就拿它与堆顶的元素进行对比。对比过程与之前相同。

堆的应用三:利用堆求中位数

中位数,即处在中间位置的那个数。如果数据的个数是奇数,把数据从小到大排列,那第 n / 2 + 1 n/2+1 n/2+1个数据就是中位数(数据从0开始编写)。如果数据是偶数的话,那处于中间位置的数据有两个,第 n / 2 n/2 n/2和第 n / 2 + 1 n/2+1 n/2+1个数据。可以随便取一个,如 n / 2 n/2 n/2为中位数。

对于一组静态数据,中位数是固定的,排序后取第 n / 2 n/2 n/2个数据。

但对于一组动态数据,中位数不断变动,再用排序的办法效率就不高了。但借助堆这种数据结构,可以非常高效地实现求中位数操作。

我们需要维护两个堆,一个大顶堆,一个小顶堆。大顶堆中存储前半部分数据,小顶堆中存储后半部分数据,且小顶堆中的数据都大于大顶堆中的数据。如下图:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-wlSyOcoV-1574951906525)(E:\数据结构(python)\images\堆9.jpg)]

其中,大顶堆的堆顶元素就是我们要找的中位数。

如果新加入的数据小于等于大顶堆的堆顶元素,我们就将这个数据插入到大顶堆中,否则,就将这个数据插入到小顶堆中。同时,通过从一个堆中不停地将堆顶元素移动到另一个堆,通过这样的调整,使其满足这样的约定。

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-spNFOcGt-1574951906526)(E:\数据结构(python)\images\堆10.jpg)]

插入数据需要堆化,故时间复杂度为O( l o g n logn logn),求中位数时,只需返回大顶堆的堆顶,时间复杂度为O(n)。

参考资料:王争《数据结构与算法之美》

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值