c++堆

c++ reference: http://www.cplusplus.com/reference/algorithm/make_heap/

  heap并不属于STL容器组件,它分为 max heap 和min heap,在缺省情况下,max-heap是优先队列(priority queue)的底层实现机制。

而这个实现机制中的max-heap实际上是以一个vector表现的完全二叉树(complete binary tree)。

  二叉堆(binary heap)就是i一种完全二叉树。也即是。整棵二叉树除了最底层的叶节点以外,都是填满的,而最低层的叶子结点必须是从左到右不留空隙。

至于max-heap和min-heap,前者的任何一个父亲结点都必须大于等于他的任意子结点,而后者相反。

 

下面我们利用数组来隐式表达这棵数:

  第0号元素保留,从arry[1]开始保存A,这时候我们可以轻易的看到:

  位于位置i的某个结点arry[i],他的左子结点必然在arry[2*i]中,右子结点必然位于arry[2*i+1],其父亲结点必然位于arry[i/2]处。

  这种数组表达的方式我们 称为 隐式表达。

  当然由于arry大小是静态的,不能动态添加元素,我们可以使用vector来实现。

heap-算法:

1. push_heap(),新添加一个元素在末尾,然后重新调整堆序。也就是把元素添加在底层vector的end()处。

该算法必须是在一个已经满足堆序的条件下,添加元素。该函数接受两个随机迭代器,分别表示first,end,区间范围。

关键是我们执行一个siftup()函数,上溯函数来重新调整堆序。具体的函数机理很简单,可以参考我的编程珠玑里面堆的实现的文章。

2. pop_heap(),这个算法跟push_heap类似,参数一样。不同的是我们把堆顶元素取出来,放到了数组或者是vector的末尾,用原来末尾元素去替代,

然后end迭代器减1,执行siftdown()下溯函数来重新调整堆序。

注意算法执行完毕后,最大的元素并没有被取走,而是放于底层容器的末尾。如果要取走,则可以使用底部容器(vector)提供的pop_back()函数。

3. sort_heap(),既然每次pop_heap可以获得堆中最大的元素,那么我们持续对整个heap做pop_heap操作,每次将操作的范围向前缩减一个元素。

当整个程序执行完毕后,我们得到一个非降的序列。

同理,sort_heap(RamdomAccessIteraor first,RamdomAccessIteraor end)接受两个随机迭代器作为参数。表示操作的范围。

注意这个排序执行的前提是,在一个堆上执行。执行完之后序列也就失去了堆的性质。

#include<iostream> 
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<vector>
#include<deque>
#include<map> 
#include<set>
#include <sstream>
using namespace std;void outHeap(vector<int> v){for(int i=0; i<v.size(); ++i)cout<<v[i]<<" ";cout<<endl;
}int main(){int myints[] = {10,20,30,5,15};vector<int> v(myints,myints+5);cout<<"建堆:"<<endl;make_heap(v.begin(), v.end());outHeap(v);cout<<endl;cout<<"往堆里插入一个元素:"<<endl;v.push_back(100);push_heap(v.begin(), v.end());outHeap(v);cout<<endl;cout<<"弹出堆顶元素,输出下一个堆顶元素:" <<endl;cout<<"当前堆顶元素: "<<v.front()<<endl;pop_heap(v.begin(), v.end());v.pop_back();cout<<"下一个堆顶元素: "<<v.front()<<endl;cout<<endl;cout<<"排序堆:"<<endl;sort_heap(v.begin(), v.end());//默认从小到大 //sort_heap(v.begin(), v.end(), greater<int>());
    outHeap(v);//通过multiset实现最小堆 cout<<endl<<"通过multiset实现最小堆:"<<endl;multiset<int> mst(myints,myints+5);for(multiset<int>::iterator it = mst.begin(); it!=mst.end(); ++it) cout<<*it<<" ";cout<<endl;return 0;
} 

 4.一道很经典的题目就是在1亿个数中找到最大的前100个数,这是一道堆应用题,找最大的前100个数,那么我们就创建一个大小为100的最小化堆,每来一个元素就与堆顶元素比较,因为堆顶元素是目前前100大数中的最小数,前来的元素如果比该元素大,那么就把原来的堆顶替换掉并重新调整堆。

5.例题:lintcode 滑动窗口的中位数 : http://www.lintcode.com/zh-cn/problem/sliding-window-median/

 

//最无脑的解法....
class
Solution { public:/*** @param nums: A list of integers.* @return: The median of the element inside the window at each moving*/vector<int> medianSlidingWindow(vector<int> &nums, int k) {vector<int>v;vector<int> res;if (k > nums.size() || k == 0) return res;for(int i=0; i<k; ++i)v.push_back(nums[i]);sort(v.begin(), v.end());res.push_back(v[(k-1)/2]);for(int i=k; i<nums.size(); ++i){v.erase(lower_bound(v.begin(), v.end(), nums[i-k]));v.insert(lower_bound(v.begin(), v.end(), nums[i]), nums[i]);res.push_back(v[(k-1)/2]);}return res;} };
//使用multiset进行优化(内部以平衡二叉树),感觉和上面的"最脑的解法差不多", 只不过是将一个有序的序列分成左右连个连续的序列,左边序列的最后一个就是中位数
class
Solution { public:/*** @param nums: A list of integers.* @return: The median of the element inside the window at each moving*/vector<int> medianSlidingWindow(vector<int> &nums, int k) {// write your code herevector<int> res;if (k > nums.size() || k == 0) return res;multiset<int> left, right;//init heaps by first kth elements in numsfor (int i = 0; i < k; ++i) {left.insert(nums[i]);}while (left.size() > (k + 1) / 2) {right.insert(*left.rbegin());left.erase(left.find(*left.rbegin()));}res.push_back(*left.rbegin());//slide windowfor (int i = k; i < nums.size(); ++i) {//delete the leftmost element in window from heapsif (nums[i-k] > res.back()) right.erase(right.find(nums[i-k]));else left.erase(left.find(nums[i-k]));//insert new element into heapsif (!left.empty() && nums[i] <= *left.rbegin()) left.insert(nums[i]);else right.insert(nums[i]);//adjust heaps so that the left heap contains (k + 1) / 2 elementsif (left.size() < (k + 1) / 2) {left.insert(*right.begin());right.erase(right.begin());} else if (left.size() > (k + 1) / 2) {right.insert(*left.rbegin());left.erase(left.find(*left.rbegin()));}res.push_back(*left.rbegin());}return res;} };

 

转载于:https://www.cnblogs.com/hujunzheng/p/5001898.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/531382.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

关于Ubuntu拒绝root用户ssh远程登录

今天使用SecureCRT远程登陆Ubuntu时一直提示密码或用户名错误&#xff0c;实际输入是正确的&#xff0c;我按照网上教程改还是不行&#xff0c;后来才想起来我是root登录的&#xff0c;Ubuntu默认的ssh远程root登录是关闭的&#xff0c;在这里记录一下 1.编辑配置文件 #sudo v…

windows8建立局域网的方法

win8建立局域网的方法&#xff1a;1、首先笔记本有无线网卡且支持 虚拟WIFI ;2、按winX键&#xff0c;选择"命令提示符(管理员)A"; 3、输入"netsh wlan set hostednetwork modeallow ssid网络名称 key我的密码" ; 4、接着输入"netsh wlan start hoste…

内核移植出现:Kernel panic - not syncing: No init found.

今天在升级SDK的时候&#xff0c;升级到kernel时遇到如题所述的问题&#xff0c;花了天时间调通&#xff0c;在这里记录一下。 报错提示&#xff1a;(当时没有记录&#xff0c;错误的提示大概如下) Kernel panic - not syncing: No init found. Try passing init option to k…

lintcode Permutation Index

题目&#xff1a;http://www.lintcode.com/zh-cn/problem/permutation-index/ 排列序号 给出一个不含重复数字的排列&#xff0c;求这些数字的所有排列按字典序排序后该排列的编号。其中&#xff0c;编号从1开始。 样例 例如&#xff0c;排列[1,2,4]是第1个排列。 思路&#xf…

32位和64位机器上C语言数据类型的大小

作为嵌入式开发的人员&#xff0c;是必须了解C语言在不同位数机器上占用的字节大小的&#xff0c;下面做下对比 不同位数平台对比&#xff1a; \16位平台32位平台64位平台char1个字节8位1个字节8位1个字节short2个字节16位2个字节16位2个字节int2个字节16位4个字节32位 4个字节…

lintcode循环数组之连续子数组求和

v 题目&#xff1a;连续子数组求和 II给定一个整数循环数组&#xff08;头尾相接&#xff09;&#xff0c;请找出一个连续的子数组&#xff0c;使得该子数组的和最大。输出答案时&#xff0c;请分别返回第一个数字和最后一个数字的值。如果多个答案&#xff0c;请返回其中任意一…

lintcode最长回文子串(Manacher算法)

题目来自lintcode, 链接&#xff1a;http://www.lintcode.com/zh-cn/problem/longest-palindromic-substring/ 最长回文子串 给出一个字符串&#xff08;假设长度最长为1000&#xff09;&#xff0c;求出它的最长回文子串&#xff0c;你可以假定只有一个满足条件的最长回文串。…

全排列总结

接触全排列已经好长时间了&#xff0c;一直没有抽空总结一下全排列的相关问题&#xff0c;下面来说一下&#xff01; 排列 一般地&#xff0c;从n个不同元素中取出m&#xff08;m≤n&#xff09;个元素&#xff0c;按照一定的顺序排成一列&#xff0c;叫做从n个元素中取出m个元…

大小端问题傻傻分不清?

先来熟悉一下概念&#xff1a; 大端&#xff1a;数据的高位数据保存在低位地址&#xff0c;数据的低位数据保存在高地址 小端&#xff1a;数据的高位数据保存在高位地址&#xff0c;数据的低位数据保存在低地址为什么会存在大小端的问题&#xff1f; 这是因为在计算机系统中&a…

n个结点,不同形态的二叉树(数目+生成)

题目链接&#xff1a; 不同的二叉查找树&#xff1a;http://www.lintcode.com/zh-cn/problem/unique-binary-search-trees/ 不同的二叉查找树 II&#xff1a;http://www.lintcode.com/zh-cn/problem/unique-binary-search-trees-ii/ 不同形态二叉树的数目&#xff1a; 样例 给出…

c++ stringstream(老好用了)

前言&#xff1a; 以前没有接触过stringstream这个类的时候&#xff0c;常用的字符串和数字转换函数就是sscanf和sprintf函数。开始的时候就觉得这两个函数应经很叼了&#xff0c;但是毕竟是属于c的。c中引入了流的概念&#xff0c;通过流来实现字符串和数字的转换方便多了。在…

mount --bind的用处

&#xff08;一&#xff09;mount --bind介绍 mount --bind的作用是将两个目录连接起来&#xff0c;例如&#xff1a;mount ---bind /dir1 /dir2 是将dir1目录挂载到dir2目录上&#xff0c;下面来实际演示一下&#xff1a; 上面的操作中首先创建了dir1 dir2两个目录&#xf…

gcc -strip编译选项的作用

从字面上来看strip的意思是脱衣服、拆卸&#xff0c;那么gcc --strip的作用大概能猜错来了。 没错就是有选择地除去行号信息、重定位信息、调试段、typchk 段、注释段、文件头以及所有或部分符号表。 一旦使用该命令&#xff0c;则很难调试文件的符号&#xff0c;因此&#x…

lintcode 落单的数(位操作)

题目1 落单的数 给出2*n 1 个的数字&#xff0c;除其中一个数字之外其他每个数字均出现两次&#xff0c;找到这个数字。 链接&#xff1a;http://www.lintcode.com/zh-cn/problem/single-number/ 样例 给出 [1,2,2,1,3,4,3]&#xff0c;返回 4 挑战 一次遍历&#xff0c;常数级…

旋转图像

旋转图像 给定一个NN的二维矩阵表示图像&#xff0c;90度顺时针旋转图像。 看个例子 算法1&#xff1a; 如上图所示&#xff0c;设一个N阶二维矩阵&#xff0c;则将矩阵从外向里可以分成N/2个圈&#xff0c;例如&#xff08;1 2 3 4 8 12 16 15 14 13 9 5&#xff09;这是最外边…

嵌入式开发板模拟器:QEMU

前两天看微信公众号时发现了一个嵌入式模拟器&#xff0c;感觉很不错&#xff0c;自己动手安装了一个&#xff0c;折腾了几天&#xff0c;下载一直是个问题&#xff0c;特此记录如下 模拟器大家应该都听说过&#xff0c;有的小伙伴打游戏也会安装模拟器&#xff0c;今天我们介绍…

gcc: weak_alias如何使用

本文主要说明weak和alias是什么和如何使用它 __attribute__是用来说明函数的属性&#xff0c;weak和alias分别是两个属性。 &#xff08;一&#xff09;强符号和弱符号&#xff1a; 强符号&#xff1a;已经初始化的全局变量和未被weak修饰的函数弱符号&#xff1a;未初始化的全…

静态Include和动态Include测试并总结

主要代码 hjzgg.css .center-div{width:auto;margin-left: 40%;margin-right: 40%;display: block;position: absolute;top:0px;left:0px; }.text-div{margin-top: 80px; }.hjzgg-div{color:transparent;font-size:20px;font-weight: bold;letter-spacing:2px;-webkit-animatio…

linux终端常用快捷键

CTRLALTT 打开终端 CTRLD 关闭终端 CTRL SHIFT "" 放大终端字体 CTRL “-” 缩小终端字体 CTRL r 查找历史命令 CTRLu 删除光标前面所有内容 CTRLw 删除光标左边的单词 CTRL k 删除光标后面的所有内容 CTRLL 清除当前屏幕内容 CTRLa 光标移到开始位置 CTRLe 光标移到…

ueditor的配置和使用

ueditor下载好之后直接复制到项目的WebContent目录下&#xff0c;并将ueditor\jsp\lib下的jar包复制或者剪切到项目的lib目录下。先看一下效果&#xff0c;如下&#xff1a; 1.文件的上传 首先在ueditor/jsp目录下找到config.json文件&#xff0c;就拿Image上传来说吧。 "…