我们这里的堆是指用来表示元素集合的一种数据结构

一个二叉树是一个堆是由堆的两个性质决定的(以小根堆为例)

1:任何节点的值都小于或等于其子节点的值

2:该二叉树最多在两层上具有叶节点,其中最底层的叶节点尽可能的靠左分布

我们可以从数学上约束这两个性质

x[i/2]<=x[i](2<=i<=n)

从这个性质我们可以定义heap(l,u):x[i/2]<=x[i](2l<=i<=u)

下面我们考虑两种情况:

1:当x[1,...n-1]是堆时,在x[n]中放置一个任何的元素可能无法产生heap(1,n)

2:当x[1...n]是一个堆时,给x[1]分配一个新值得到heap(2,n),如何得到heap(1,n)

现在我们来探讨这两个问题:

x[n]即为图中带圈的元素,该元素一直往上与父节点交换,知道该节点的值大于等于其父节点(或者位于树根)为止

如果过程中heap(1,n-1)为真,那么heap(1,n)为真

void siftup(int n)//sift 筛选
{
int i=n;
while(true) {
if(i==) break;
int p=i/;
if(x[p]<=x[i]) break;
swap(i,p);
i=p;
}
}

x[1]即为图中带圈的元素,x[1]向下筛选,直到它没有子节点或者小于等于它的子节点

void siftdown(int n)
{
int i=;
while(true) {
int c=*i;
if(c>n) break;//no child node
if(c+<=n)//c+1 is the right child,c+1<=n,i has right child
if(x[c+]<x[c]) c++;//select lesser one
//c is the lesser child
if(x[i]<=x[c]) break;
swap(c,p);
i=c;
}
}

通过解决这两个问题,我们实现了给堆增加一个元素或者删除最小的元素(最小堆)

现在我们考虑用堆的这两个操作来实现另一个数据结构--优先级队列

优先级队列(priority queue) 是0个或多个元素的集合,每个元素都有一个优先权,对优先级队列执行的操作有(1)查找(2)插入一个新元素 (3)删除

一般情况下,查找操作用来搜索优先权最大的元素,删除操作用来删除该元素 。对于优先权相同的元素,可按先进先出次序处理或按任意优先权进行。

如果用我们上面讨论的最小堆来实现的话,那么对应的操作如下:

查找:返回x[1]

插入:插入新元素是将n+1,然后将新元素放在x[n]处,这样我们就具备了调用siftup的前提:heap(1,n-1)

删除:由于是最小堆,x[1]即为优先权最高的元素,删除x[1],我们可以将x[n]移动到x[1]并将n-1,这样集合中的元素都在x[1..n]中了,并且heap(2,n)为真,那我们就可以调用siftdown了

template<class T>
class priqueue
{
private:
int n,maxsize;
T *x;
void swap(int i,int j){ T t = x[i];x[i]=x[j];x[j]=t;}
public:
priqueue(int m)
{
maxsize=m;
x= new int[maxsize+];
n=;
}
void insert(T t)
{
int i,p;
x[++n]=t;
for(i=n;i> && x[p=i/]>x[i];i=p) swap(i,p);
}
T extractmin()
{
int i,c;
T t = x[];
x[]=x[n--];
for(i=;(c=*i)<=n;i=c)
{
if(c+<=n&&x[c+]<x[c])
c++;
if(x[i]<=x[c]) break;
swap(i,c);
}
return t;
}
~priqueue(); };

优先级队列的实现为我们提供了一种简单的向量排序算法:首先在优先级队列中依次插入每个元素,然后排序删除它们

然而这种想法需要额外的一个数组的内存,但我们分析可知,是可以在一个数组上操作的,想法如下:

第一阶段:在原数组上建立堆,

第二阶段:使用堆来建立有序序列,由于x[1]是最小的元素,交换x[1]和x[n--],然后把新的顶部元素向下筛选来重新获得堆性质,循环。

不过这样得到的是一个降序的有序序列,如果要得到升序的有序序列,我们要使用最大堆而不是最小堆

下面是实现的最小堆排序的代码:

#include <iostream>
#include "stdlib.h"
#include "time.h"
using namespace std;
int x[]= {};
int co=;
void swap(int f,int s)
{
int temp=x[f];x[f]=x[s];x[s]=temp;
co++;
}
void siftup(int n)//sift 筛选
{
int i=n;
while(true)
{
if(i==) break;
int p=i/;
if(x[p]<=x[i]) break;
swap(i,p);
i=p;
}
}
void siftdown(int n)
{
int i=;
while(true) {
int c=*i;
if(c>n) break;//no child node
if(c+<=n)//c+1 is the right child,c+1<=n,i has right child
if(x[c+]<x[c]) c++;//select lesser one
//c is the lesser child
if(x[i]<=x[c]) break;
swap(c,i);
i=c;
}
}
int main(int argc, char const *argv[])
{
srand(unsigned(time(NULL)));
for(int i=;i<;i++)
{
x[i]=rand()%;
cout<<x[i]<<" ";
}
cout<<endl;
//sort
for(int i=;i<=;i++) siftup(i);
for(int i=;i>=;i--)
{
swap(i,);
siftdown(i-);
}
for(int i=;i<;i++)
{
cout<<x[i]<<" ";
}
cout<<endl<<co;
return ;
}

堆--LogN的数据结构的更多相关文章

  1. java对象在JVM堆中的数据结构

    java对象和数组是存放在堆中的,那么这些instance的数据结构是什么呢? 对象头:对象头存放的是这个对象的一些元数据信息.例如每个对象都有哈希值,GC分代年龄,锁状态标志等,这些信息就是存放在对 ...

  2. 数据结构-堆 Java实现

    数据结构-堆 Java实现. 实现堆自动增长 /** * 数据结构-堆. 自动增长 * */ public class Heap<T extends Comparable> { priva ...

  3. 从一个集合中查找最大最小的N个元素——Python heapq 堆数据结构

    Top N问题在搜索引擎.推荐系统领域应用很广, 如果用我们较为常见的语言,如C.C++.Java等,代码量至少也得五行,但是用Python的话,只用一个函数就能搞定,只需引入heapq(堆队列)这个 ...

  4. 数据结构(栈&堆 )

    在计算机领域,堆栈是一个不容忽视的概念,堆栈是两种数据结构.堆栈都是一种数据项按序排列的数据结构,只能在一端(称为栈顶(top))对数据项进行插入和删除.在单片机应用中,堆栈是个特殊的存储区,主要功能 ...

  5. python数据结构与算法

    最近忙着准备各种笔试的东西,主要看什么数据结构啊,算法啦,balahbalah啊,以前一直就没看过这些,就挑了本简单的<啊哈算法>入门,不过里面的数据结构和算法都是用C语言写的,而自己对p ...

  6. C#堆栈和托管堆

    首先堆栈和堆(托管堆)都在进程的虚拟内存中.(在32位处理器上每个进程的虚拟内存为4GB) 堆栈stack 堆栈中存储值类型. 堆栈实际上是向下填充,即由高内存地址指向低内存地址填充. 堆栈的工作方式 ...

  7. (十一)C语言中内存堆和栈的区别

    在计算机领域,堆栈是一个不容忽视的概念,我们编写的C语言程序基本上都要用到.但对于很多的初学着来说,堆栈是一个很模糊的概念. 堆栈:一种数据结构.一个在程序运行时用于存放的地方,这可能是很多初学者的认 ...

  8. C++数据结构之Stack(栈)

    stack,栈,是好比堆积木似的数据结构,从上之下堆积,取出时按"LIFO"-last int first out后进先出的规则.栈一般为线程所独有,也就是每个线程有其自有的栈,与 ...

  9. C# 数据结构--排序[下]

    希尔排序(Shell Sort) 排序思想: 先取一个小于n的整数d1作为第一个增量,把文件的全部记录分组.所有距离为d1的倍数的记录放在同一个组中.先在各组内进行直接插入排序:然后,取第二个增量d2 ...

随机推荐

  1. C#基础-连接Access与SQL Server

    1.连接Access数据库 string strConnection = "Provider=Microsoft.Ace.OleDb.12.0; Data Source=" + S ...

  2. POJ3348:Cows——题解

    http://poj.org/problem?id=3348 题目大意:用已给出的点围出面积最大的凸包,输出面积/50(向下取整) —————————————————————————— 第一道凸包?以 ...

  3. BZOJ3144:[HNOI2013]切糕——题解

    https://www.lydsy.com/JudgeOnline/problem.php?id=3144 看着很像网络流,但是费用流貌似无法解决这个问题,其实甚至连忽略d的情况都做不到. 最小割? ...

  4. 【算法】【网络流24题】巨坑待填(成功TJ,有时间再填)

    ------------------------------------------------------------------------------------ 17/24 --------- ...

  5. bzoj3157: 国王奇遇记

    emmm...... 直接看题解好了: BZOJ-3157. 国王奇遇记 – Miskcoo's Space O(m)不懂扔掉 总之,给我们另一个处理复杂求和的方法: 找到函数之间的递推公式! 这里用 ...

  6. python基础之魔法方法

    由于hexo自带的markdown渲染引擎对双下划线做了转义,在正文中看到的魔法方法前后都没有双下划线 setattr.getattr.delattr 可以拦截对对象属性的访问 setattr函数是用 ...

  7. PHP 数据库防止攻击

    定义和用法 mysql_real_escape_string() 函数转义 SQL 语句中使用的字符串中的特殊字符. 下列字符受影响: \x00 \n \r \ ' " \x1a 如果成功, ...

  8. [转]Android 如何根据网络地址获取网络图片方法

    http://blog.csdn.net/xiazdong/article/details/7724103 目录(?)[-] h2pre namecode classhtml stylefont-we ...

  9. B树及其变种

    B树是为磁盘或其他直接存取的辅助存储设备而设计的一种平衡搜索树.B树类似于红黑树,但它们在降低磁盘I/O操场数方面要更好一些.许多数据库系统使用B树或B树的变种来存储信息. 介绍 常见的动态查找树包括 ...

  10. linux删除乱码

    一:前沿 好久没有记载东西了,回来之后一直忙着改东西,我走之前项目是什么样,回来后也差不多.郁闷啊,努力敲代码,但是要敲出思想来啊.先会做,然后深入思考 二:内容 (1)每次使用rz命令向服务器上传代 ...