Priority_queue
一、priority_queue的介绍和使用
1.1 priority_queue的介绍
1.优先队列是一种容器适配器,根据严格的弱排序标准,它的第一个元素总是它所包含的元素中最大的。
2.优先队列类似于堆, 在堆中可以随时插入元素, 并且只能检索最大堆元素(优先队列中位于顶部的元素)。
3.优先队列被实现为容器适配器,容器适配器即将特定的容器类封装作为其底层容器,queue提供一组特定的成员函数来访问其元素,元素从特定容器的尾部弹出,这里称为优先队列的顶部。
4.底层容器可以是任何标准容器类模版,也可以是其他特定设计的容器类,容器应该可以通过随机访问和迭代器访问,需要支持一下操作:
-
empty() : 判断容器是否为空
-
size() : 返回容器中的有效元素个数
-
front() : 返回容器中的第一个元素的引用
-
push_back() : 在容器中尾插元素
-
pop_back() : 在容器中尾删元素
5.标准容器类vector和deque满足这些需求,默认情况下如果没有指定底层容器的话就默认使用vector作为优先队列的底层容器。
6.需要支持随机访问迭代器,以便始终在内部保持堆结构。容器适配器通过在需要时自动调用算法函数make_heap、push_heap、和pop_heap来自动完成此操作。
1.2 priority_queue的使用
优先级队列默认使用vector作为求底层存储数据的容器,在vector上又使用了堆算法将vector中元素构造成堆的结构,因此priority_queue就是堆,所有需要用到堆的位置,都可以使用priority_queue来替代。
注意:默认情况下,priority_queue是大堆。
| 函数声明 | 接口说明 |
|---|---|
| priority_queue()/priority_queue(first, last) | 构造一个空的优先级队列 |
| empty() | 检测优先级队列是否为空,是的话就返回true否则返回false |
| top() | 返回优先级队列中最大(最小)的元素,即堆顶元素(默认是大堆,返回最大元素) |
| push(x) | 在优先级队列中插入元素x |
| pop() | 删除优先级队列中最大(最小)元素,即删除堆顶元素 |
下面我们来看下优先队列的使用实例:
#include<iostream>
#include<vector>
#include<queue>
//greater算法的头文件
#include<functional>using namespace std;void test1()
{//默认情况下是建大堆priority_queue<int> q1;vector<int> arr{ 1, 3, 4, 5, 6 , 7, 9, };for (auto& e : arr){q1.push(e);}while (!q1.empty()){cout << q1.top() << " ";q1.pop();}cout << endl;//将第三个模版参数换成great比较方式就是建小堆priority_queue<int, vector<int>, greater<int>> q2(arr.begin(), arr.end());while (!q2.empty()){cout << q2.top() << " ";q2.pop();}cout << endl;}int main()
{test1();return 0;
}
二、OJ练习
leetcode215,数组中的第k个最大元素
题目:

我们先看题意是让我们找出数组nums中的第k个最大的元素,看到样例2中我们发现相同的数也会被算进k里面,因此我们在这题上面,就可以利用我们的优先级队列,我们将优先级队列中的前k -1 个元素给pop出去,那么剩下的元素中堆顶元素就是我们的第k个最大元素。
下面是这道题的参考代码:
class Solution {
public:int findKthLargest(vector<int>& nums, int k) {priority_queue<int> q1(nums.begin(), nums.end());//将前k - 1个元素pop掉就是第k个最大元素while(--k) q1.pop();//直接返回优先级队列的堆顶元素即可return q1.top();}
};
三、 priority_queue的模拟实现
3.1 仿函数的概念
仿函数又称为函数对象:重载了operator()的类,类的对象可以向函数一样使用。
operator() 的特点:参数个数和返回值是根据需求定的很灵活。
3.2 仿函数的实现
3.2.1 判断小于的仿函数
下面是我们的判断小于的仿函数的代码:
//判断小于的仿函数
template<class T>
class myless
{
public:bool operator() (T& x, T& y){return x < y;}
};
我们定义了一个重载了operator() 的类,类里面就只有一个重载函数。我们可以通过这种方式来模拟出一个函数。
3.2.2 判断大于的仿函数
下面是我们判断大于的仿函数的代码:
//判断大于的仿函数
template<class T>
class mygreater
{
public:bool operator() (T& x, T& y){return x > y;}
};
这是我们判断大于的仿函数,和判断小于的仿函数的原理相同也是使用了一个类模版,然后再里面重载了一个operator() 函数,用来模拟函数的效果。
3.3 构造以及建堆调整操作
3.3.1 构造函数
这里的构造函数我们先实现默认的无参构造:
//无参构造
priority_queue():c(), comp()
{}
将类里面的两个成员变量给一个空值, c是底层容器,comp是用来比较的仿函数。实现建堆的调整函数后我们在来实现我们的迭代器构造函数。
3.3.2 向上调整建堆
下面是我们进行维持优先级队列的顺序的函数之一,向上调整算法:
//向上调整建堆
void Adjust_up(int child)
{int parent = (child - 1) / 2;while (child > 0){//if (c[parent] < c[child])if(comp(c[parent], c[child])){swap(c[parent], c[child]);}child = parent;parent = (child - 1) / 2;}
}
我们首先要知道用数组模拟堆,也就是优先级队列,它的子节点和父亲节点的下标之间的关系,在向上建堆操作中我们是从下往上的,我们需要知道parent和child之间的关系,它们之间的关系就是 parent = (child - 1) / 2 , 这里的比较我们可以使用仿函数来比较也可以直接实用大于和小于号比较但是建议使用仿函数,因为仿函数的通用性高, 当我们的父节点小于我们的子节点时我们就将他们交换位置,然后将父节点的位置给到子节点,然后重新计算父节点,相当于是更新子节点和父节点。
3.3.3 向下调整建堆
下面是我们的向下调整算法:
//向下调整建堆
void Adjust_down(int parent)
{int child = parent * 2 + 1;if (child + 1 < c.size() && c[child] < c[child + 1]){child++;}while (child < c.size()){//if (c[parent] < c[child])if(comp(c[parent], c[child])){swap(c[parent], c[child]);}parent = child;child = parent * 2 + 1;}
}
向下建堆的效率要比向上调整建堆的效率要高很多, 在我们的迭代器区间构造和我们的pop函数中可以使用我们的向下调整算法来维持优先队列的结构。原理和向上调整建堆基本相同,向下调整算法是传入parent的值,然后计算出child的值,child和parent的关系式是:child = parent * 2 + 1, 然后我们选出最大的左右子节点中较大的那个子节点,对parent和child进行比较,如果parent的值小于child的值的话就将它们调换位置,然后将child的下标给给parent,然后再计算child的下标,相当于更新位置。
3.3.4 使用迭代器区间进行构造
下面是我们的实例代码:
//迭代器区间构造template <class InputIterator>priority_queue(InputIterator first, InputIterator last):c(),comp(){while (first != last){c.push_back(*first);first++;}for (int i = (c.size() - 1 - 1) / 2; i >= 0; i--){Adjust_down(i);}}
这里我们直接将这个迭代器区间中的值通过迭代器遍历逐个放进我们的底层容器中即可,然后通过我们的向下调整算法进行建堆。
3.4 priority_queue中的容量以及访问访问修改操作
3.4.1 容量相关的接口
我们容量相关的接口有size和empty,size的作用是返回有效数据的个数,empty的作用是判断队列中是否为空。
下面来看我们的size接口:
//返回有效数据个数
size_t size() const
{return c.size();
}
对于我们的size接口就直接返回我们底层容器的size接口的值即可。
再来看下我们的empty接口:
//判断优先队列里面是否为空bool empty() const{return c.empty();}
与size接口同理empty接口也是直接返回我们的底层容器的接口即可。
3.4.2 访问和修改
关于访问和修改操作的接口有:top, push, pop
下面依次来看这些接口:
top:
//返回队头元素T& top() {return c[0];}const T& top() const{return c[0];}
top 接口的作用是返回我们的队头元素,我们的底层容器是动态数组vector我们就直接返回数组的第一个元素即可。
push:
//入队操作
//尾插入队,执行向上调整操作
void push(const T& x)
{c.push_back(x);Adjust_up(c.size() - 1);
}
对于入队操作我们直接尾插进数组即可,尾插之后执行向上调整操作。
pop:
//出队操作
//先交换最后一个元素和第一个元素的位置
//尾删出队,执行向下调整操作
void pop()
{swap(c[0], c[size() - 1]);c.pop_back();Adjust_down(0);
}
对于我们pop接口的作用时将堆顶元素给pop出去,我们将堆顶元素和数组最后一个元素进行交换位置,然后进行尾删,最后执行向下调整操作,维持堆的结构即可。
这就是这篇文章的全部内容了,希望大家能从以上内容中对优先级队列有一定的理解。
对于入队操作我们直接尾插进数组即可,尾插之后执行向上调整操作。pop:```C++
//出队操作
//先交换最后一个元素和第一个元素的位置
//尾删出队,执行向下调整操作
void pop()
{swap(c[0], c[size() - 1]);c.pop_back();Adjust_down(0);
}
对于我们pop接口的作用时将堆顶元素给pop出去,我们将堆顶元素和数组最后一个元素进行交换位置,然后进行尾删,最后执行向下调整操作,维持堆的结构即可。
这就是这篇文章的全部内容了,希望大家能从以上内容中对优先级队列有一定的理解。
相关文章:
Priority_queue
一、priority_queue的介绍和使用 1.1 priority_queue的介绍 1.优先队列是一种容器适配器,根据严格的弱排序标准,它的第一个元素总是它所包含的元素中最大的。 2.优先队列类似于堆, 在堆中可以随时插入元素, 并且只能检索最大堆…...
SpringMVC:获取请求数据
1. 通过RequestParma注解接收 /**** value和name都可以使用,互为别名* 如果此处设置了需要什么参数而前端请求时没有提供则会报400(请求参数不一致错误)* required参数用于设置该参数是否为必须传递参数,默认为true必须传递* defa…...
深度学习 --- stanford cs231 编程作业(assignment1,Q2: SVM分类器)
stanford cs231 编程作业之SVM分类器 写在最前面: 深度学习,或者是广义上的任何学习,都是“行千里路”胜过“读万卷书”的学识。这两天光是学了斯坦福cs231n的一些基础理论,越往后学越觉得没什么。但听的云里雾里的地方也越来越多…...
【scikit-learn010】sklearn算法模型清单实战及经验总结(已更新)
1.一直以来想写下基于scikit-learn训练AI算法的系列文章,作为较火的机器学习框架,也是日常项目开发中常用的一款工具,最近刚好挤时间梳理、总结下这块儿的知识体系。 2.熟悉、梳理、总结下scikit-learn框架模型算法包相关技术点及经验。 3.欢迎批评指正,欢迎互三,跪谢一键…...
Rethinking overlooked aspects in vision-language models
探讨多模态视觉语言模型的一些有趣结论欢迎关注 CVHub!https://mp.weixin.qq.com/s/zouNu-g-33_7JoX3Uscxtw1.Introduction 多模态模型架构上的变化不大,数据的差距比较大,输入分辨率和输入llm的视觉token大小是比较关键的,适配器,VIT和语言模型则不是那么关键。InternVL-…...
【漯河市人才交流中心_登录安全分析报告-Ajax泄漏滑动距离导致安全隐患】
前言 由于网站注册入口容易被黑客攻击,存在如下安全问题: 暴力破解密码,造成用户信息泄露短信盗刷的安全问题,影响业务及导致用户投诉带来经济损失,尤其是后付费客户,风险巨大,造成亏损无底洞…...
C语言—字符函数和字符串函数
1.字符分类函数 C语言中有一系列的函数是专门做字符分类的,也就是一个字符是属于什么类型的字符的。 这些函数的使用都需要包含一个头文件 ctype.h。 例:将一句话中的小写字母改成大写字母。 2.字符转换函数 头文件:ctype.h C语言提供了2…...
爬山算法的详细介绍
爬山算法(Hill Climbing Algorithm)是一种基于启发式的局部搜索算法,常用于解决优化问题。它的核心思想是从当前解的邻域中选择能够使目标函数值最大(或最小)的下一个解作为当前解,直到找到一个满足问题要求…...
硕士课程 可穿戴设备之作业一
作业一 第一个代码使用的方法是出自于[1]。 框架结构 如下图,不过根据对代码的解读,发现作者在代码中省去了对SSR部件的实现,下文再说。 Troika框架由三个关键部件组成:信号分解,SSR和光谱峰值跟踪。(粗…...
测试记录3:WLS2运行Linux界面
1.WLS1转到WLS2 (1)根据自己的平台,下载WLS2安装包 x64: https://wslstorestorage.blob.core.windows.net/wslblob/wsl_update_x64.msi arm64: https://wslstorestorage.blob.core.windows.net/wslblob/wsl_update_arm64.msi (2&…...
好用软件推荐
软件功能相关介绍地址FastStone截图(长截图、定时截图等)CSDNhttps://www.faststone.org/FSCaptureDownload.htmQuicker快捷访问https://getquicker.net/https://getquicker.net/...
王学岗鸿蒙开发(北向)——————(二)TS基本语法详解
1,Ts(TypeScript)语法相当于JAVAScript类型,鸿蒙arkTs是基于TS语言的,当然artTs也融合了其它的语言。 2,本篇文章是基于n9版本。注意,有些语法是已经不能用的。 3, 4,变量:用来存储数据,数字字母组成,数字不…...
【网络协议 | HTTP】HTTP总结与全梳理(一) —— HTTP协议超详细教程
🔥博客简介:开了几个专栏,针对 Linux 和 rtos 系统,嵌入式开发和音视频开发,结合多年工作经验,跟大家分享交流嵌入式软硬件技术、音视频技术的干货。 ✍️系列专栏:C/C、Linux、rtos、嵌入式…...
java基础选择题--11
1. 以下保留字( )不能出现在说明虚函数原型的语句中。A.static B.operator C.void D.const 参考答案:A 2. 一个类中只能定义一个析构函数。( )A.对 B.错 参考答案:A 解释: 在C中,一个类只能有一个析构函数。析构函数在对象生…...
欲除烦恼须无我,各有前因莫羡人
欲除烦恼须无我,各有前因莫羡人...
Vue的APP实现下载文件功能,并将文件保存到手机中
Vue的APP实现下载文件功能,并将文件保存到手机中 文字说明后台核心代码前台核心代码运行截图项目链接 文字说明 本文介绍Vue实现的APP,将文件下载并保存到手机中,为系统提供导出功能;同时支持导入,即选择本地的文件后&…...
泛微开发修炼之旅--07通过后端代码实现创建并发送待办、源码及示例
文章链接:泛微开发修炼之旅--07通过后端代码实现创建并发送待办、源码及示例...
轻松搭建AI应用的三个大模型技术路线
时下聊起AI,想必最热的就是使用AI的应用(chatGPT,文心一言等)来提升自己工作的效率,比如破局俱乐部,洋哥带领星球2万多人开启大航海,教人使用这一波新起的应用进行赚钱与赋能。 在我的视角来看…...
Vue01-vue的简介
一、Vue是什么? 一套用于构建用户界面的渐进式javaScript框架。 构建用户界面: 渐进式: 目前Vue的地位:生态完善,国内前端工程师必备技能。 二、Vue的特点 一个XXX.vue就是一个组件,封装的概念,…...
leetcode455.分发饼干、376. 摆动序列、53. 最大子序和
455.分发饼干 为了满足更多的小孩,就不要造成饼干尺寸的浪费 大尺寸的饼干既可以满足胃口大的孩子也可以满足胃口小的孩子,那么就应该优先满足胃口大的 这里的局部最优就是大饼干喂给胃口大的,充分利用饼干尺寸喂饱一个,全局最…...
大数据学习栈记——Neo4j的安装与使用
本文介绍图数据库Neofj的安装与使用,操作系统:Ubuntu24.04,Neofj版本:2025.04.0。 Apt安装 Neofj可以进行官网安装:Neo4j Deployment Center - Graph Database & Analytics 我这里安装是添加软件源的方法 最新版…...
Xshell远程连接Kali(默认 | 私钥)Note版
前言:xshell远程连接,私钥连接和常规默认连接 任务一 开启ssh服务 service ssh status //查看ssh服务状态 service ssh start //开启ssh服务 update-rc.d ssh enable //开启自启动ssh服务 任务二 修改配置文件 vi /etc/ssh/ssh_config //第一…...
JavaScript 中的 ES|QL:利用 Apache Arrow 工具
作者:来自 Elastic Jeffrey Rengifo 学习如何将 ES|QL 与 JavaScript 的 Apache Arrow 客户端工具一起使用。 想获得 Elastic 认证吗?了解下一期 Elasticsearch Engineer 培训的时间吧! Elasticsearch 拥有众多新功能,助你为自己…...
FFmpeg 低延迟同屏方案
引言 在实时互动需求激增的当下,无论是在线教育中的师生同屏演示、远程办公的屏幕共享协作,还是游戏直播的画面实时传输,低延迟同屏已成为保障用户体验的核心指标。FFmpeg 作为一款功能强大的多媒体框架,凭借其灵活的编解码、数据…...
Java-41 深入浅出 Spring - 声明式事务的支持 事务配置 XML模式 XML+注解模式
点一下关注吧!!!非常感谢!!持续更新!!! 🚀 AI篇持续更新中!(长期更新) 目前2025年06月05日更新到: AI炼丹日志-28 - Aud…...
JVM暂停(Stop-The-World,STW)的原因分类及对应排查方案
JVM暂停(Stop-The-World,STW)的完整原因分类及对应排查方案,结合JVM运行机制和常见故障场景整理而成: 一、GC相关暂停 1. 安全点(Safepoint)阻塞 现象:JVM暂停但无GC日志,日志显示No GCs detected。原因:JVM等待所有线程进入安全点(如…...
回溯算法学习
一、电话号码的字母组合 import java.util.ArrayList; import java.util.List;import javax.management.loading.PrivateClassLoader;public class letterCombinations {private static final String[] KEYPAD {"", //0"", //1"abc", //2"…...
【分享】推荐一些办公小工具
1、PDF 在线转换 https://smallpdf.com/cn/pdf-tools 推荐理由:大部分的转换软件需要收费,要么功能不齐全,而开会员又用不了几次浪费钱,借用别人的又不安全。 这个网站它不需要登录或下载安装。而且提供的免费功能就能满足日常…...
R 语言科研绘图第 55 期 --- 网络图-聚类
在发表科研论文的过程中,科研绘图是必不可少的,一张好看的图形会是文章很大的加分项。 为了便于使用,本系列文章介绍的所有绘图都已收录到了 sciRplot 项目中,获取方式: R 语言科研绘图模板 --- sciRplothttps://mp.…...
【Linux系统】Linux环境变量:系统配置的隐形指挥官
。# Linux系列 文章目录 前言一、环境变量的概念二、常见的环境变量三、环境变量特点及其相关指令3.1 环境变量的全局性3.2、环境变量的生命周期 四、环境变量的组织方式五、C语言对环境变量的操作5.1 设置环境变量:setenv5.2 删除环境变量:unsetenv5.3 遍历所有环境…...
