数据结构:堆

堆性质

堆是一种特别的树状数据结构。若满足以下特性,即可称为堆:“给定堆中任意节点P和C,若P是C的父节点,那么P的值会小于等于(或大于等于)C的值”。若父节点的值恒小于等于子节点的值,此堆称为最小堆;反之,称为最大堆。

通常最用的是二叉堆,为二叉树的一种,有以下性质:

  • 任意节点小于等于(或大于等于)它的子节点,最小元(或最大元)在堆的根上。
  • 堆总是一颗完全树。即除了最底层,其他层的节点都被元素填满,且最底层从左到右填入。

数据结构:堆_第1张图片

可以看到,二叉堆是一种部分排序,只对父子节点的排序做了限制,对左右子节点的大小排序则没有做限制。我们看一下堆的具体实现。

堆的实现

因为堆总是一颗完全树,所以其存储可以用数组(变长数组)来表示。用数组表示的非常大的好处是能快速定位一个节点在堆中的位置。根节点在数组中的下标可以从0开始,也可以从1开始,这里从1开始。当一个元素在数组中的位置为 i i i时,其左子节点 l e f t = 2 i left = 2i left=2i,右子节点 r i g h t = 2 i + 1 right = 2i + 1 right=2i+1,相应的其父节点 p a r e n t = [ i / 2 ] parent = [i / 2] parent=[i/2]([]表示取整)。这个是可以推导出来的。

推导思路如下:
其实上面三个公式,只要推导出 l e f t = 2 i left = 2i left=2i,其他两个很自然就推导出来了。而推导左子节点的思路,假定节点 i i i在堆的第 l l l层的第 j j j个节点,有 l e f t = 2 ( l − 1 ) + j − 1 left = 2^{(l-1)} + j-1 left=2(l1)+j1, 则左子节点 s u b l e f t sub_{left} subleft就在堆的第 l + 1 l+1 l+1层的第 2 ( j − 1 ) + 1 − 1 2(j-1) + 1 - 1 2(j1)+11的位置上,即 s u b l e f t = 2 l + 2 ( j − 1 ) sub_{left} = 2^l + 2(j-1) subleft=2l+2(j1),可以看到 s u b l e f t = 2 ∗ l e f t sub_{left} = 2 * left subleft=2left,推导完毕。

插入

插入的实现思路: 主要从堆的性质去考虑怎么实现,堆总是一颗完全树,所以,新插入的节点为满足完全树的性质要放到树叶子节点空闲的最左节点,但这样可能不满足父节点总大于等于子节点的要求,所以,还要判断,如果大于父节点,就上移,直到满足小于父节点的条件。算法描述如下:

  1. Add the element to the bottom level of the heap at the most left.
  2. Compare the added element with its parent; if they are in the correct order, stop.
  3. If not, swap the element with its parent and return to the previous step.

时间复杂度,最坏情况 O ( l o g ( n ) ) O(log(n)) O(log(n)),最好情况很简单就是 O ( 1 ) O(1) O(1)

删除根节点

删除根节点的实现思路: 与插入类似,堆要删除一个节点(根节点),还要满足是完全树,只能是“删除”最右测的叶子节点,将该节点复制到根节点,但此时虽满足完全树却不满足父节点大于子节点的性质,因此,要不断将该节点下移,直到满足父节点大于子节点。算法描述如下:

  1. Replace the root of the heap with the last element on the last level.
  2. Compare the new root with its children; if they are in the correct order, stop.
  3. If not, swap the element with one of its children and return to the previous step. (Swap with its smaller child in a min-heap and its larger child in a max-heap.)

可以看到无论是加入节点还是删除节点,都是先满足堆的完全二叉树的性质,再去调整其父节点与子节点的大小关系,而堆底层的数据结构是动态数组或者说是向量,直接在向量尾部进行增删操作效率最高且容易实现。在代码实现时,最主要的就是上移和下移操作,较为容易实现。

堆的构造

构造一个堆,最简单的方法就是一个元素一个元素的插入到堆中,在最坏的情况下,插入n个元素的时间复杂度为 l o g ( 1 ) + l o g ( 2 ) + . . . + l o g ( n ) = l o g ( 1 ∗ 2 ∗ . . . ∗ n ) = l o g ( n ! ) = O ( n l o g ( n ) ) log(1) + log(2) + ... + log(n) = log(1 * 2 * ... * n) = log(n!)=O(nlog(n)) log(1)+log(2)+...+log(n)=log(12...n)=log(n!)=O(nlog(n))。这种方法也被称为是自顶向下式的,还有另一种自底向上构造的方法,可以将一个数组转换为堆,时间复杂度可以做到 O ( n ) O(n) O(n)

具体方法是从一个节点元素任意放置的二叉树开始(表示形式为一个没有堆性质的数组),自底向上堆每个子树执行“下移”操作,先构造较小的堆,再将他们重复合并成很大的堆。

下移操作(即下移元素i直到满足父节点大于等于子节点这个堆性质)算法描述如下:

Max-Heapify (A, i):
    left = 2*i
    right = 2*i+1
    largest = i

    if left <= heap_length[A] and A[left] > A[largest] then:
        largest = left
    
    if right <= heap_length[A] and A[right] > A[largest] then:
        largest = right
    
    if largest != i then:
        swap(A[i], A[largest])
        Max-Heapify (A, largest)

构造堆的算法描述如下:

Build-Max-Heap (A):
    heap_length[A] = length[A]
    // 这块其实就是对二叉树中所有非叶子节点进行“下移”操作,从最底层向最上构造堆
    for each index i from floor(length[A]/2) downto 1 do:
        Max-Heapify (A, i) 

这个构造堆的算法的时间复杂度分析比较复杂一些,具体的分析可参考Building a heap,这里说一下分析的思路:分析这个时间复杂度,我们知道一次“下移”操作的时间复杂度是 O ( h ) O(h) O(h),时间复杂度与树高度有关,然后分析每一层的时间复杂度,再分析每一层最多有多少个节点,最后将所有层累加起来就得到了,式子比较复杂,还需要级数化简等步骤。

堆的应用

说道堆的应用,最常见的就是优先队列的实现了,把堆稍微封装一下就可以实现优先队列了。见代码heap.h。其他的还有堆排序,在图论中等等其他方面的应用,这里不再列出。


欢迎关注公众号:一日思考,获取更多数据结构与算法的内容。
数据结构:堆_第2张图片

你可能感兴趣的:(数据结构与算法)