【C++】priority_queues(优先级队列)和反向迭代器适配器的实现

目录

  • 一、 priority_queue
    • 1.priority_queue的介绍
    • 2.priority_queue的使用
      • 2.1、接口使用说明
      • 2.2、优先级队列的使用样例
    • 3.priority_queue的底层实现
      • 3.1、库里面关于priority_queue的定义
      • 3.2、仿函数
        • 1.什么是仿函数?
        • 2.仿函数样例
      • 3.3、实现优先级队列
        • 1. 1.0版本的实现
        • 2. 2.0版本的实现
  • 二、反向迭代器适配器

前言

继上一篇stack和queue我们讲解了其实现原理,里面也提到了容器适配器的概念,本篇我们要讲的优先级队列,也是一种容器适配器,另外我们再顺带讲一下反向迭代器,这个也是一个容器适配器哦,废话不多说,我们直接切入正题

一、 priority_queue

1.priority_queue的介绍

priority_queue他是一种容器适配器,但其实他底层和堆差不多,接口和堆也非常像,功能也是,默认情况下是大堆,你也可以用仿函数把他改成小堆

它的接口有以下几个:

  1. empty():检测容器是否为空
  2. size():返回容器中有效元素个数
  3. front():返回容器中第一个元素的引用
  4. push_back():在容器尾部插入元素
  5. pop_back():删除容器尾部元素

priority_queue的底层是堆,堆其实是完全二叉树,而完全二叉树的物理结构又是类似数组这种连续的物理空间,所以说适配priority_queue的容器要能够随机访问下标,需要支持随机访问迭代器,以便始终在内部保持堆结构,一般我们用vector作为它的默认容器,deque也可以

2.priority_queue的使用

2.1、接口使用说明

在这里插入图片描述

2.2、优先级队列的使用样例

priority_queue<int> pq;
pq.push(1);
pq.push(2);
pq.push(3);
pq.push(4);
pq.push(5);
while (!pq.empty())
{cout << pq.top() << " ";pq.pop();
}
//打印结果是5,4,3,2,1

tips:默认情况下大的优先级高,底层是个大堆

3.priority_queue的底层实现

3.1、库里面关于priority_queue的定义

在这里插入图片描述
priority_queue类模板参数多了一个Compare,这个参数是用来调节大小堆的,默认的less是大堆,greater是小堆
tips:
在这里插入图片描述

3.2、仿函数

1.什么是仿函数?

仿函数又被叫做函数对象,它们是通过重载operator()运算符的类的实例,它们可以像函数那样被调用,具有这样特性的就是仿函数

2.仿函数样例
template<class T>
struct Less
{bool operator()(const T& x, const T& y){return x < y;}
};int main()
{Less<int> lessfunc;cout << lessfunc.operator()(1, 2) << endl;cout << lessfunc(2, 3) << endl;//就这样乍一看还以为是函数调用,其实这是仿函数cout << Less<int>()(1, 2) << endl;//通过匿名对象来调用return 0;
}

3.3、实现优先级队列

1. 1.0版本的实现
template<class T,class Container=vector<T>>
class priority_queue
{
public:size_t size(){return _con.size();}void adjust_up(size_t child){size_t parent = (child - 1) / 2;while (child>0){if (_con[child] > _con[parent]){swap(_con[child], _con[parent]);child = parent;parent = (child - 1) / 2;}else{break;}}}void adjust_down(size_t parent){size_t child = parent * 2 + 1;while (child<_con.size()){if (child + 1 <_con.size() && _con[child] < _con[child + 1]){child++;}if (_con[child] > _con[parent]){swap(_con[child], _con[parent]);parent = child;child = parent * 2 + 1;}else{break;}}}void push(const T& val){_con.push_back(val);//先尾插adjust_up(_con.size()-1);//再向上调整}void pop(){swap(_con[0], _con[_con.size() - 1]);//先把要删除的堆顶元素和最后一个元素交换_con.pop_back();//然后删除最后一个元素adjust_down(0);//再进行向下调整}const T& top(){return _con[0];}bool empty(){return _con.empty();}private:Container _con;
};

这里重点讲一下向上调整建堆和向下调整建堆,我们以建小堆为例:
在这里插入图片描述
向下调整的原理和向上调整很像,我就不多讲解了

2. 2.0版本的实现
template<class T>
struct less//这个虽然叫less但是它是大堆
{bool operator()(const T& x, const T& y){return x < y;}
};
template<class T>
struct greater//这个虽然叫greater,但是他是小堆
{bool operator()(const T& x, const T& y){return x > y;}
};
template<class T,class Container=vector<T>,class Com=less<T>>
class priority_queue
{
public:size_t size(){return _con.size();}void adjust_up(size_t child){Com com;//搞一个仿函数对象size_t parent = (child - 1) / 2;while (child>0){//if (_con[child] > _con[parent])//if ( _con[parent]<_con[child] )if(com(_con[parent],_con[child])){//注意这里换成仿函数的时候要和它里面的<对上,再替换成仿函数对象调用swap(_con[child], _con[parent]);child = parent;parent = (child - 1) / 2;}else{break;}}}void adjust_down(size_t parent){Com com;size_t child = parent * 2 + 1;while (child<_con.size()){//if (child + 1 <_con.size() && _con[child] < _con[child + 1])if (child + 1 < _con.size() && com(_con[child] , _con[child + 1])){child++;}//if (_con[child] > _con[parent])//if (_con[parent]< _con[child])if (com(_con[parent] , _con[child])){swap(_con[child], _con[parent]);parent = child;child = parent * 2 + 1;}else{break;}}}void push(const T& val){_con.push_back(val);//先尾插adjust_up(_con.size()-1);//再向上调整}void pop(){swap(_con[0], _con[_con.size() - 1]);//先把要删除的堆顶元素和最后一个元素交换_con.pop_back();//然后删除最后一个元素adjust_down(0);//再进行向下调整}const T& top(){return _con[0];}bool empty(){return _con.empty();}private:Container _con;
};

tips:

int main()
{priority_queue<int,vector<int>,greater<int>> pq;//注意这里:如果你要传仿函数的参数类型,一定不要忘记了这个vector<int>//不能跳过这个缺省参数去传他后面的其他参数,切记切记!!!return 0;
}

二、反向迭代器适配器

反向迭代器适配器,可以根据正向迭代器适配出它相应的反向迭代器

反向迭代器的实现思想其实很简单,相比我们前面list的实现;我们在这里实现反向迭代器主要是利用正向迭代器来替我们完成,库里面的实现讲求了对称,begin/end和rbegin/rend是堆成的
在这里插入图片描述

template<class iterator, class Ref, class Ptr>
struct ReserveIterator
{typedef ReserveIterator<iterator, Ref, Ptr> Self;iterator _it;ReserveIterator(iterator it):_it(it){}Ref operator*(){Iterator tmp = _it;return *(--tmp);}Ptr operator->(){return &(operator*());}Self& operator++(){--_it;return *this;}Self& operator--(){++_it;return *this;}bool operator!=(const Self& s){return _it != s._it;}
};

关于容器适配器之类的容器我们就先讲到这里,我们下期浅谈一下模板✌

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

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

相关文章

android自定义view仿微信联系人列表

说明&#xff1a;最近碰到一个需求&#xff0c;弄一个类似国家或省份列表&#xff0c;样式参照微信联系人 文件列表&#xff1a; step1:主界面 加载列表数据~\app\src\main\java\com\example\iosdialogdemo\MainActivity.java step2:右侧列表数据排序~\app\src\com\example\io…

6. 第K小的和-二分

6.第K小的和 - 蓝桥云课 (lanqiao.cn) #include <bits/stdc.h> #define int long long #define endl \n using namespace std; int n,m,k,an[100005],bm[100005]; int check(int x){int res0;//序列C中<x的数的个数for(int i0;i<n;i){//遍历数组A&#xff0c;对于每…

神级框架!!不要再封装各种 Util 工具类了【送源码】

这个工具类就比较厉害了&#xff0c;不过我在 Halo 当中用得最多的还是 HtmlUtil.encode&#xff0c;可以将一些字符转化为安全字符&#xff0c;防止 xss 注入和 SQL 注入&#xff0c;比如下面的评论提交。 comment.setCommentAuthor(HtmlUtil.encode(comment.getCommentAutho…

汇聚荣科技:拼多多开店没有流量应该怎么办?

拼多多开店没有流量是一个常见的问题&#xff0c;许多新手商家都会遇到这样的困境。那么&#xff0c;如何解决这个问题呢?下面从四个方面进行详细阐述。 一、优化店铺和商品 首先&#xff0c;要确保店铺和商品的质量。店铺要有自己独特的风格和特色&#xff0c;商品要有高质量…

Allegro如何输出各层PCB视图的PDF文件

如何输出各层PCB视图的PDF文件 1、说明 用Allegro设计好PCB后&#xff0c;有时需要出各层的PDF文档出来进行汇报和展示&#xff0c;这时就需要将各层的平面视图全部以PDF的形式加载出来&#xff0c;具体方法如下。 2、PDF文件的输出方法&#xff08;以四层板为例&#xff09; …

原子学习笔记7——FrameBuffer 应用编程

Frame 是帧的意思&#xff0c;buffer 是缓冲的意思&#xff0c;所以 Framebuffer 就是帧缓冲&#xff0c;这意味着 Framebuffer 就是一块内存&#xff0c;里面保存着一帧图像。 应用程序通过对 LCD 设备节点/dev/fb0&#xff08;假设 LCD 对应的设备节点是/dev/fb0&#xff09;…

css如何实现边框模糊的效果

其实并不难&#xff0c;用属性 filter: blur(数字px); 即可。效果如下&#xff1a; 图上的圆形内有色彩的渐变&#xff0c;同样也是用filter: blur(数字px); 实现的&#xff0c;代码如下&#xff1a;、 <template><div id"root" :style"{}">…

ros键盘控制程序teleop_twist_keyboard 键值含义及用法

在机器人仿真中&#xff0c; 经常会用到键盘控制程序teleop_twist_keyboard 对机器人进行控制。但是对各个键值是何种含义&#xff0c; 如何操作并没有任何资料介绍,初次使用时会不知所措。 通过实践&#xff0c; 发现各个键值的作用如下&#xff1a; u-- 向左前方前进 i-- 直…

RIP动态路由协议详解

目录 一&#xff1a;RIP协议的基本信息 二&#xff1a;RIP协议中的更新方式 三&#xff1a;RIP协议中的计时器 定时更新器&#xff08;UPDATE timer&#xff09; 无效定时器&#xff08;invalid Timer&#xff09; 垃圾收集定时器&#xff08;garbage collection timer&a…

第五课,输入函数、布尔类型、比较运算和if判断

一&#xff0c;输入函数input() 与输出函数print()相对应的&#xff0c;是输入函数input()&#xff0c;前者是把程序中的数据展示给外界&#xff08;比如电脑屏幕上&#xff09;&#xff0c;而后者是把外界&#xff08;比如键盘&#xff09;的数据输入进程序中 input()函数可…

Verilog代码bug:一种特殊的组合逻辑环

Verilog代码bug&#xff1a;一种特殊的组合逻辑环 组合逻辑环&#xff08;Combinational Loop&#xff09;是什么&#xff0c;别的文章已经写的很多了&#xff0c;本文就不赘述了&#xff0c;本文主要记录遇到过的一种特殊的逻辑环&#xff1b; 代码如下所示&#xff1a; mo…

MacApp自动化测试之Automator初体验

今天我们继续讲Automator的使用。 初体验 启动Automator程序&#xff0c;选择【工作流程】类型。从资源库区域依次将获取指定的URL、从网页中获得文本、新建文本文件三个操作拖进工作流创建区域。 然后修改内容&#xff0c;将获取指定的URL操作中的URL替换成https://www.cnb…

for循环 while循环

for循环 for循环格式 for 变量 in 取值列表 #for in &#xffe5; &#xff08;seq 1 10&#xff09; do 命令序列 .......... done 另一种 for &#xff08;&#xff08;变量初始值&#xff1b; 变量范围&#xff0c; 变量迭代方…

JDK1.8 安装并配置环境变量

一、Windows 配置 1 安装文件 jdk-8u401-windows-i586.exe 2 环境变量 JAVA_HOME C:\Program Files (x86)\Java\jdk-1.8 CLASSPATH .;%JAVA_HOME%\lib\tools.jar;%JAVA_HOME%\lib\dt.jar; Path %JAVA_HOME%\bin 说明&#xff1a;Win7/Win8 中 Path 可能需要写成 ;%JAVA_HO…

Edge浏览器自动翻译功能按钮不见了

前言&#xff1a; 平时偶尔会用到Edge的页面翻译功能&#xff0c;使用挺方便。突然发现Edge浏览器的翻译功能不见 了。如下图所示&#xff1a; 解决思路&#xff1a; 1、从网上找各种解决方案也没有解决&#xff0c;其中有一个说到点右上角的三个点 2、点击设置…

30W 宽电压输入 AC/DC 导轨式开关电源——TPR/DG-30-XS 系列

TPR/DG-30-XS 系列导轨式开关电源&#xff0c;额定输出功率为30W&#xff0c;产品输入范围&#xff1a;90-264VAC。提供12V、24V输出&#xff0c;12V输出时&#xff0c;工作温度范围 (-25℃~70℃)具有短路保护&#xff0c;过载保护等功能&#xff0c;并具备高效率&#xff0c;高…

Windows内核--Kernel API简析(3.1)

如果所有的内核提供的功能&#xff0c;内核提供进程/线程创建和终止&#xff0c;内存分配和释放&#xff0c;文件操作&#xff0c;网络功能&#xff0c;驱动程序加载和卸载等功能。这些API将在后面陆续介绍&#xff0c;如下先介绍Kernel提供的基础API(Kernel自身或Driver使用).…

视频号小店,一个不用直播就可以变现的项目!创业首选!

大家好&#xff0c;我是电商小V 想要创业或者是想要利用视频号变现的小伙伴可以说是很多的&#xff0c;因为视频号这两年的流量是非常大的&#xff0c;甚至即将超越抖音的流量&#xff0c;因为视频号背靠腾讯平台&#xff0c;也是不缺少流量的&#xff0c;并且视频号的流量是可…

实时“秒回”,像真人一样语音聊天,GPT-4o模型强到恐怖

今天凌晨OpenAl发布了 GPT-4o&#xff0c;这是一种新的人工智能模式&#xff0c;集合了文本、图片、视频、语音的全能模型。 能实时响应用户的需求&#xff0c;并通过语音来实时回答你&#xff0c;你可以随时打断它。还具有视觉能力&#xff0c;能识别物体并根据视觉做出快速的…

6、Qt—Log4Qt使用小记1

开发平台&#xff1a;Win10 64位 开发环境&#xff1a;Qt Creator 13.0.0 构建环境&#xff1a;Qt 5.15.2 MSVC2019 64位 一、Log4Qt简介 Log4Qt是使用Trolltech Qt Framework的Apache Software Foundation Log4j包的C 端口。它旨在供开源和商业Qt项目使用。所以 Log4Qt 是Apa…