二叉堆可以看做一个近似的完全二叉树,所以一般用数组来组织。

二叉堆可以分为两种形式:最大堆和最小堆。最大堆顾名思义,它的每个结点的值不能超过其父结点的值,因此堆中最大元素存放在根结点中。最小堆的组织方式刚好与最大堆相反,它的最小元素存放在根结点中。

维护堆性质最重要的两个算法就是向上维护和向下维护。简而言之,例如最大堆中根结点的值小于其子结点的值,这个时候就要向下维护,把根结点逐级下降到适合的位置。显而易见地,向上维护就是子结点的值比其父结点大时(最大堆中),将结点逐级上升到合适的位置。这两个方法保证堆的性质不会被破坏。

堆经常用来实现优先队列,最大堆就对应最大优先队列,最小堆同上。因为通过关键字来查找堆中具体元素的位置比较麻烦,所以一般通过在堆中存储对象的句柄,在对应的对象中也存储对应堆元素的句柄来直接定位到元素的位置。

二叉堆是堆中最容易实现的一种,也算是用的比较广泛的一种。我下面的代码给出了最大堆和最小堆的一部分关键操作,另外一部分则挑选其中一种来实现。

代码如下:(仅供参考)

 class Node {
public :
int value;
public :
Node(int v = ) : value(v) {}
}; class Heap {
vector<Node> heap;
int heap_size; int Left(int i) {return (i << ) + ;} //下标从0开始
int Right(int i) {return Left(i) + ;}
int Parent(int i) {return (i - ) >> ;}
void MaxHeapify(int i); //使结点i维持最大堆性质(向下维护)
void MinHeapify(int i); //使结点i维持最小堆性质(向下维护)
void IncreaseKey(int i, int k); //将结点i的value上升到k(向上维护)
public :
Heap() : heap_size() {}
Heap(vector<Node> &t) : heap(t), heap_size(t.size()) {}
void BuildMaxHeap(); //建立最大堆,时间复杂度O(n);
void BuildMinHeap(); //建立最小堆,时间复杂度O(n);
void MinHeapSort(); //从小到大排序,时间复杂度O(nlgn);
void MaxHeapSort(); //从大到小排序,时间复杂度O(nlgn);
//以下函数仅在最大堆下进行
void Insert(int k); //插入一个元素(最大堆)
void Delete(int i); //删除一个元素(最大堆)
Node Maximum(); //返回value最大的元素
Node ExtractMax(); //去掉并返回value最大的元素
//union two heap:union the vector of the two heaps, and call BuildMaxHeap()
int Size() {return heap.size();} //返回vector的元素个数
void PrintAll() {
for (auto i : heap)
cout << i.value << ends;
cout << endl;
}
}; void Heap::MaxHeapify(const int i) {
int largest;
if (Left(i) < heap_size && heap[Left(i)].value > heap[i].value)
largest = Left(i);
else
largest = i;
if (Right(i) < heap_size && heap[Right(i)].value > heap[largest].value)
largest = Right(i);
if (largest != i) {
swap(heap[i], heap[largest]);
MaxHeapify(largest);
}
} void Heap::MinHeapify(const int i) {
int least;
if (Left(i) < heap_size && heap[Left(i)].value < heap[i].value)
least = Left(i);
else
least = i;
if (Right(i) < heap_size && heap[Right(i)].value < heap[least].value)
least = Right(i);
if (least != i) {
swap(heap[i], heap[least]);
MinHeapify(least);
}
} void Heap::BuildMaxHeap() {
heap_size = heap.size();
for (int i = Parent(heap_size - ); i >= ; --i)
MaxHeapify(i);
} void Heap::BuildMinHeap() {
heap_size = heap.size();
for (int i = Parent(heap_size - ); i >= ; --i)
MinHeapify(i);
} void Heap::MinHeapSort() {
BuildMaxHeap();
for (int i = heap.size() - ; i > ; --i) {
swap(heap[i], heap[]);
--heap_size;
MaxHeapify();
}
} void Heap::MaxHeapSort() {
BuildMinHeap();
for (int i = heap.size() - ; i > ; --i) {
swap(heap[i], heap[]);
--heap_size;
MinHeapify();
}
} void Heap::Insert(int k) {
heap.push_back(INT_MIN);
IncreaseKey(heap.size() - , k);
} void Heap::Delete(int i) {
if (heap[heap.size() - ].value > heap[i].value) {
IncreaseKey(i, heap[heap.size() - ].value);
heap.pop_back();
} else {
heap[i] = heap[heap.size() - ];
heap.pop_back();
heap_size = heap.size();
MaxHeapify(i);
}
} Node Heap::Maximum() {
return heap[];
} Node Heap::ExtractMax() {
Node max = heap[];
heap[] = heap[heap.size() - ];
heap.pop_back();
MaxHeapify();
return max;
} void Heap::IncreaseKey(int i, int k) {
if (k <= heap[i].value)
return ;
while (i > && heap[Parent(i)].value < k) {
heap[i] = heap[Parent(i)];
i = Parent(i);
}
heap[i].value = k;
}

binary-heap(二叉堆)原理及C++代码实现的更多相关文章

  1. Binary Heap(二叉堆) - 堆排序

    这篇的主题主要是Heapsort(堆排序),下一篇ADT数据结构随笔再谈谈 - 优先队列(堆). 首先,我们先来了解一点与堆相关的东西.堆可以实现优先队列(Priority Queue),看到队列,我 ...

  2. 【数据结构与算法Python版学习笔记】树——利用二叉堆实现优先级队列

    概念 队列有一个重要的变体,叫作优先级队列. 和队列一样,优先级队列从头部移除元素,不过元素的逻辑顺序是由优先级决定的. 优先级最高的元素在最前,优先级最低的元素在最后. 实现优先级队列的经典方法是使 ...

  3. 堆(Heap)和二叉堆(Binary heap)

    堆(Heap) The operations commonly performed with a heap are: create-heap: create an empty heap heapify ...

  4. 二叉堆(binary heap)

    堆(heap) 亦被称为:优先队列(priority queue),是计算机科学中一类特殊的数据结构的统称.堆通常是一个可以被看做一棵树的数组对象.在队列中,调度程序反复提取队列中第一个作业并运行,因 ...

  5. 二叉堆(binary heap)—— 优先队列的实现

    二叉堆因为对应着一棵完全二叉树,因而可以通过线性数组的方式实现. 注意,数组第 0 个位置上的元素,作为根,还是第 1 个位置上的元素作为根? 本文给出的实现,以数组第 1 个位置上的元素作为根,则其 ...

  6. 数据结构 之 二叉堆(Heap)

    注:本节主要讨论最大堆(最小堆同理). 一.堆的概念     堆,又称二叉堆.同二叉查找树一样,堆也有两个性质,即结构性和堆序性.     1.结构性质:     堆是一棵被完全填满的二叉树,有可能的 ...

  7. C# 实现简单的 Heap 堆(二叉堆)

    如题,C#  实现简单的二叉堆的 Push() 和 Pop(), 如有不足欢迎指正. 另外,在C#中使用 Heap 的相似功能可以考虑使用:Priority Queues,SortedDictiona ...

  8. 【nodejs原理&源码杂记(8)】Timer模块与基于二叉堆的定时器

    [摘要] timers模块部分源码和定时器原理 示例代码托管在:http://www.github.com/dashnowords/blogs 一.概述 Timer模块相关的逻辑较为复杂,不仅包含Ja ...

  9. 【nodejs原理&源码杂记(8)】Timer模块与基于二叉堆的定时器

    目录 一.概述 二. 数据结构 2.1 链表 2.2 二叉堆 三. 从setTimeout理解Timer模块源码 3.1 timers.js中的定义 3.2 Timeout类定义 3.3 active ...

随机推荐

  1. python np array转json

    np array转json import numpy as np import codecs, json a = np.arange().reshape(,) # a by array b = a.t ...

  2. idea~创建maven webapp项目

    1.选择 org.apache.maven.archtypes:maven-archtype-webapp 2.禁止远程下载 archetypeCatalog=internal 目的是不远程下载,否则 ...

  3. Day 5 :ArrayList原理、LinkedList原理和方法和迭代器注意事项

    迭代器在变量元素的时候要注意事项: 在迭代器迭代元素 的过程中,不允许使用集合对象改变集合中的元素个数,如果需要添加或者删除只能使用迭代器的方法进行操作.   如果使用过了集合对象改变集合中元素个数那 ...

  4. 常用函数式接口与Stream API简单讲解

    常用函数式接口与Stream API简单讲解 Stream简直不要太好使啊!!! 常用函数式接口 Supplier<T>,主要方法:T get(),这是一个生产者,可以提供一个T对象. C ...

  5. [Qt5] QSlider设置步长

    这是一个小问题,就是QSlider是一个滑动条控件,既然是个滑动条控件,就会想要用鼠标滚轮或者鼠标去移动它来实现某些功能,但是呢,我能说这个控件的一个属性函数设置也是比较奇怪的,它设置步长的函数有 s ...

  6. 24.docker 部署 wordPress

    1. 拉取远程 mysql 和 wordpress 镜像 并 启动起来 使用 docker pull mysql:5.7.27 docker pull wordpress 2. 创建mysql 的 c ...

  7. 【WPF学习】第三十七章 触发器

    WPF中有个主题,就是以声明方式扩展代码的功能.当使用样式.资源或数据绑定时,将发现即使不使用代码,也能完成不少工作. 触发器是另一个实现这种功能的例子.使用触发器,可自动完成简单的样式改变,而这通常 ...

  8. bfs--奇怪的电梯P1135

    计院有一个bug电梯,可能是hyk造的,很多bug,电梯只有两个按钮,“上”和“下”,电梯每层都可以停,每层都有一个数字Ki(0<=Ki<=n),当你在一层楼,你按“上”键会到1+K1层, ...

  9. 设计模式讲解2:static proxy和decorator的不同点

    声明:迁移自本人CSDN博客https://blog.csdn.net/u013365635 在常见的23种设计模式中,static proxy和decorator在代码结构上是特别相似的.那它们的不 ...

  10. 邮件发送,无尽的501错误。TCP发送邮件解决方案

    先贴上错误信息,便于搜索引擎采集,也送给遇到此问题的技术朋友们. smtp 501 Syntax error (no parameters allowed) (#5.5.4) 背景描述: 使用TCP发 ...