C++ 模拟实现 priority_queue(优先队列)

目录

一,优先队列简介

二,priority_queue 的内部实现原理

三,模拟实现 priority_queue

1,模板参数与数据结构

2,构造

3,辅助功能(堆的有序化,建立堆)

4,核心功能

四,简单的测试 priority_queue 与完整代码


一,优先队列简介

1,什么是优先队列:

优先队列是一种特殊的队列数据结构,它具有队列先进先出(FIFO)的特性,但是在取出元素时会优先取出具有最高优先级的元素。这意味着插入的元素会按照一定的优先级规则进行排序,而不是简单地按照插入顺序。

2,STL中的优先队列:

在 C++ 的 STL 中,priority_queue 是一个容器适配器,用于实现优先队列的功能。priority_queue 提供了插入元素、移除最高优先级元素、访问堆顶元素等常用操作,操作十分便捷。

3,优先队列的使用场景:

① 任务调度:在操作系统中,任务调度往往需要根据各个任务的优先级来决定执行顺序,优先队列可以很好地满足这种场景。
② 事件处理:在网络编程或并发编程中,事件处理的顺序可能会影响程序性能,优先队列可以帮助我们按照一定的规则处理事件顺序。
③ 一些图论算法:很多图论算法(例如 Dijkstra 算法)中都需要根据权值来确定优先级,这时候可以使用优先队列来保存待处理的顶点,并按照权值进行排序。

优先队列示意图:

二,priority_queue 的内部实现原理

1,堆结构

优先队列通常是基于二叉堆来实现的,二叉堆能够很好的实现优先队列的基本操作(为方便表述,接下来将二叉堆简称为堆)。那么,什么是堆呢?

堆(Heap)是一种特殊的树形数据结构,根据不同的排列方式通常可以分为最大堆和最小堆两种类型。以最大堆为例:它的特点是父节点的键值总是大于或等于任何一个子节点的键值。这种性质决定了堆的根节点(也就是堆顶)的键值是最大的。

堆通常是一个完全二叉树,除了最底层,其它层全部都会被填充满,最底层从左到右填充。我们用数组就可以很方便地表示一个堆,而且堆的操作也多是通过数组的索引进行的。

二叉堆示意图:

2,二叉堆的表示

如果我们用指针来表示堆结构,那么每个元素都需要左子节点,右子节点,父节点三个指针;而如果我们使用数组来表示,就会变得特别方便,不需要指针就可以沿着树上下的移动。具体方式如下:arr[0] 可以用来当作根节点,对于 arr[k],它的左右两个子节点的索引分别为 2k + 1 ,2k + 2,它的父节点的索引则为 (k - 1) / 2。

(有些表示方法用 arr[1] 来作为根节点,而 arr[0] 则用来作为哨兵闲置)

用数组表示堆的示意图:

三,模拟实现 priority_queue

priority_queue 在 STL 中以底部的容器来完成几乎全部的工作,所以它并没有被归类为容器(container),而是被归类为容器适配器(container adapter)。

priority_queue 只有队列中的顶部元素(优先级最高的元素)才能被外界取用,所以 priority_queue 不提供遍历功能,这也代表着它并不提供迭代器

1,模板参数与数据结构

template<class T, class Compare = std::less<T>, class Container = std::vector<T>>
class priority_queue {
private:Container _cont;	// 底层容器Compare _cmp;		// 比较方式// ...
}

前面有说到我们会用数组来作为底层容器来表示堆,为了方便底层容器的扩容,这里选择缺省使用 vector 来作为底层容器。

2,构造

public:// 构造priority_queue() {}template<class input_iterator>priority_queue(input_iterator begin, input_iterator end) : _cont(begin, end) {make_heap();}priority_queue(const std::initializer_list<T>& lt) :_cont(lt) {make_heap();}

make_heap() 函数会将 _cont 中的元素给调整成一个堆,具体实现稍后就会给出。

3,辅助功能(堆的有序化,建立堆)

首先是寻找父节点与子节点的索引功能:

private:size_t parent_idx(size_t idx) { return (idx - 1) / 2; }size_t left_child_idx(size_t idx) { return 2 * idx + 1; }size_t right_child_idx(size_t idx) { return 2 * idx + 2; }

我们对优先队列进行一些操作时会破环堆的结构,之后我们会遍历堆,将堆的结构给恢复回来。这个恢复的过程就叫做堆的有序化

堆的有序化分为两种情况:

① 当某个节点的优先级上升,或者是我们在堆底(底层容器的尾部)加入了一个新的元素时,我们需要自下而上的对堆进行调整。

② 当某个节点的优先级下降时,我们需要自上而下的对堆进行调整。

先来看看自下而上的调整:

private:void adjust_up_heap(size_t child) {size_t parent = parent_idx(child);while (child != 0 and _cmp(_cont[parent], _cont[child])) {				std::swap(_cont[parent], _cont[child]);child = parent;parent = parent_idx(child);							}}

如果堆的结构因为某个节点比它的父节点优先级更高而被打破,那么我们就需要交换当前的节点与父节点。不过交换之后可能还会在父节点处继续打破堆的结构,所以我们要将父节点更新为当前节点,继续进行交换,直到堆恢复完成。

再来看看自上而下的调整:

private:void adjust_down_heap(size_t parent) {size_t n = _cont.size();size_t child = left_child_idx(parent);while (child < n) {if (child + 1 < n and _cmp(_cont[child], _cont[child + 1])) {++child;	// 将 left_child 变为 right_child}if (_cmp(_cont[parent], _cont[child])) {std::swap(_cont[parent], _cont[child]);parent = child;child = left_child_idx(parent);}else break;}}

如果堆的结构因为某个节点比它的子节点优先级更小而被打破,那么我们就需要将它和它的两个子节点中优先级更高的那一位进行交换。同样的,交换之后子节点处可能还会破环堆的结构,所以我们需要将子节点更新为当前节点,继续进行交换,直到堆恢复完成。

引用《算法 (第四版)》 中的一段话:

如果我们把堆想象成一个严密的黑社会组织,每个子结点都表示一个下属(父结点则表示它的直接上级),那么这些操作就可以得到很有趣的解释。adjust_up_heap() 表示一个很有能力的新人加入组织并被逐级提升(将能力不够的上级踩在脚下),直到他遇到了一个更强的领导。adjust_donw_heap() 则类似于整个社团的领导退休并被外来者取代之后,如果他的下属比他更厉害,他们的角色就会交换,这种交换会持续下去直到他的能力比其下属都强为止。

最后我们来看看怎么将一个无序的容器调整成一个堆:

private:void make_heap() {size_t tail_child = _cont.size() - 1;size_t parent = parent_idx(tail_child);size_t stop = -1;while (parent != stop) {adjust_down_heap(parent--);}}

我们从最尾部的元素的父节点开始向上遍历,每次都将当前的节点进行向下调整操作。

4,核心功能

priority_queue 的核心功能如下:插入元素(push),删除顶部元素(pop),获取顶部元素(top),检查优先队列是否为空(empty),获取元素数量(size)。

public:// 核心功能void push(const T& data) {_cont.push_back(data);adjust_up_heap(_cont.size() - 1);}void pop() {_cont.front() = _cont.back();_cont.pop_back();adjust_down_heap(0);}T& top() { return _cont.front(); }size_t size() { return _cont.size(); }bool empty() { return _cont.empty(); }void swap(const priority_queue& pq) {std::swap(_cont, pq._cont);std::swap(_cmp, pq._cmp);}

四,简单的测试 priority_queue 与完整代码

1,简单的对 priority_queue 进行测试

我们可以大量的对 priority_queue 进行插入,然后再一个一个的取出顶部的元素。

void test_priority_queue() {std::vector<int> vec;for (int i = 0;i < 100;++i) {vec.push_back(rand() % 60);}mySTL::priority_queue<int>heap(vec.begin(), vec.end());heap.push(100);while (not heap.empty()) {cout << heap.top() << " ";heap.pop();}
}

运行的结果:

2,完整代码

namespace mySTL {template<class T, class Compare = std::less<T>, class Container = std::vector<T>>class priority_queue {private:Container _cont;	// 底层容器Compare _cmp;		// 比较方式public:// 构造priority_queue() {}template<class input_iterator>priority_queue(input_iterator begin, input_iterator end) : _cont(begin, end) {make_heap();}priority_queue(const std::initializer_list<T>& lt) :_cont(lt) {make_heap();}public:// 核心功能void push(const T& data) {_cont.push_back(data);adjust_up_heap(_cont.size() - 1);}void pop() {_cont.front() = _cont.back();_cont.pop_back();adjust_down_heap(0);}T& top() { return _cont.front(); }size_t size() { return _cont.size(); }bool empty() { return _cont.empty(); }void swap(const priority_queue& pq) {std::swap(_cont, pq._cont);std::swap(_cmp, pq._cmp);}private:size_t parent_idx(size_t idx) { return (idx - 1) / 2; }size_t left_child_idx(size_t idx) { return 2 * idx + 1; }size_t right_child_idx(size_t idx) { return 2 * idx + 2; }private:void make_heap() {size_t tail_child = _cont.size() - 1;size_t parent = parent_idx(tail_child);size_t stop = -1;while (parent != stop) {adjust_down_heap(parent--);}}void adjust_down_heap(size_t parent) {size_t n = _cont.size();size_t child = left_child_idx(parent);while (child < n) {if (child + 1 < n and _cmp(_cont[child], _cont[child + 1])) {++child;	// 将 left_child 变为 right_child}if (_cmp(_cont[parent], _cont[child])) {std::swap(_cont[parent], _cont[child]);parent = child;child = left_child_idx(parent);}else break;}}void adjust_up_heap(size_t child) {size_t parent = parent_idx(child);while (child != 0 and _cmp(_cont[parent], _cont[child])) {				std::swap(_cont[parent], _cont[child]);child = parent;parent = parent_idx(child);							}}//public://	// 方便调试//	void print() {//		for (const auto& data : _cont) {//			std::cout << data << " ";//		}//		std::cout << std::endl;//	}};}

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

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

202012青少年软件编程(Python)等级考试试卷(一级)

第 1 题 【单选题】 运行下方代码段&#xff0c;输出是6&#xff0c;则输入的可能是&#xff08; &#xff09;。 a eval(input())print(a)A :8%2 B :8/2 C :3*2 D :3**2 正确答案:C 试题解析: 第 2 题 【单选题】 关于Python变量&#xff0c;下列叙述正确的是&#x…

【智能算法】鹦鹉优化算法(WO)原理及实现

目录 1.背景2.算法原理2.1算法思想2.2算法过程 3.结果展示4.参考文献 1.背景 2024年&#xff0c;J Lian等人受到鹦鹉学习行为启发&#xff0c;提出了鹦鹉优化算法&#xff08;Parrot Optimizer, PO&#xff09;。 2.算法原理 2.1算法思想 PO灵感来自于在驯养的鹦鹉中观察到的…

八皇后问题-使用递归回溯方法用c语言实现

学习的视频&#xff1a;懒猫老师 https://www.bilibili.com/video/BV1wJ411U7Gy/?spm_id_from333.337.search-card.all.click&vd_sourceda60f9e1bc3321cae28c29fe80e9b078 代码&#xff1a; 编译&#xff1a;gcc test.c -g #include <stdio.h> #include<st…

【openLooKeng集成Hive连接器完整过程】

【openLooKeng集成Hive连接器完整过程】 一、摘要二、正文2.1 环境说明2.2 Hadoop安装2.2.1. 准备工作2.2.2 在协调节点coordinator上进行安装hadoop2.2.3、将Hadoop安装目录分发到从节点worker2.2.4、在协调节点coordinator上启动hadoop集群2.3 MySQL安装2.4 Hive安装及基本操…

数据结构算法——链表带环问题——数学深度解析

前言:本节内容主要是讲解链表的两个问题 &#xff1a;1、判断链表是否带环&#xff1b; 2、一个链表有环&#xff0c; 找到环的入口点。 本节内容适合正在学习链表或者链表基础薄弱的友友们哦。 我们先将问题抛出来&#xff0c;友友们可以自己去力扣或者牛客网去找相应题目&…

el-tabs作为子组件使用页面空白

文章目录 前言一、问题展示二、源码分析三、解决方案 前言 如果el-tabs是子组件&#xff0c;父组件传值value / v-model为空字符&#xff0c;这个时候在watch中监听value / v-model就会发现监听的数据会被调用为‘0’。一定是作为子组件引用&#xff0c;且在watch进行监听&…

【Java探索之旅】包管理精粹 Java中包的概念与实践

文章目录 &#x1f4d1;前言一、封装1.1 封装的概念1.2 访问限定修饰符 二、封装扩展&#xff08;包&#xff09;2.1 包的概念2.2 带入包中的类2.3 自定义包2.4 常见的包 &#x1f324;️全篇总结 &#x1f4d1;前言 在Java编程中&#xff0c;封装是面向对象编程的核心概念之一…

令牌技术详解

1. 问题引出 之前我们讲 Cookie 和 Session 时提到过一个用户登录的场景&#xff1a;当用户登录时&#xff0c;服务器端可以把用户的登录信息存在Session中 并返回给客户端对应的SessionID&#xff0c;客户端会把这个SessionID存在Cookie 中当下次访问该服务器时&#xff0c;…

【C语言/数据结构】经典链表OJ习题~第二期——链中寻环

&#x1f388;&#x1f388;&#x1f388;欢迎采访小残风的博客主页&#xff1a;残风也想永存-CSDN博客&#x1f388;&#x1f388;&#x1f388; &#x1f388;&#x1f388;&#x1f388;本人码云 链接&#xff1a;残风也想永存 (FSRMWK) - Gitee.com&#x1f388;&#x1f…

JUC并发-共享模型-不可变

1、日期转换的问题 下面的代码在运行时&#xff0c;由于 SimpleDateFormat 不是线程安全的 Slf4j(topic "c.Test1") public class Test1 {public static void main(String[] args) {SimpleDateFormat sdf new SimpleDateFormat("yyyy-MM-dd");for (int…

Mac系统常用操作

文章目录 1、常用快捷键2、常用功能及其操作 1、常用快捷键 win和mac键盘对比&#xff1a;Command按键和Ctrl按键类似&#xff0c; 图片来源&#xff1a;https://www.xiaohongshu.com/explore/62d2787a0000000011012ab9锁屏&#xff1a;ControlCommandQ复制、粘贴、剪切、全选…

hdc不是内部或外部命令,也不是可运行的程序或批处理文件。【鸿蒙报错已解决】

文章目录 项目场景:问题描述原因分析:解决方案:此Bug解决方案总结Bug解决方案寄语项目场景: 最近遇到了这个问题,看到网上也有人在询问这个问题,实操了很多网上的解决方案发现并不能解决这个Bug,所以我在解决这个问题后,总结了自己和其他人的解决经验,进行了整理,写…

linux系统的rsync命令实现本机到远程主机之间目录的复制和同步

一、rsync命令介绍 在Linux中&#xff0c;rsync 是一个强大的命令行工具&#xff0c;用于同步文件和目录。它可以在本地或通过网络在远程系统之间复制文件。 二、远程目录复制的条件 1、系统要已经安装rsync工具 要使用 rsync 复制远程目录&#xff0c;需要确保系统上安装了 …

知识图谱与知识表示:人工智能的基石

知识图谱与知识表示&#xff1a;人工智能的基石 一、知识图谱&#xff1a;连接数据的桥梁1.1 知识图谱的构成1.2 知识图谱的应用 二、知识表示&#xff1a;AI的推理基础2.1 知识表示的定义2.2 知识表示的形式 三、从符号表示到向量表示3.1 符号表示与向量表示3.2 向量表示的优势…

自动化机器学习——网格搜索法:寻找最佳超参数组合

自动化机器学习——网格搜索法&#xff1a;寻找最佳超参数组合 在机器学习中&#xff0c;选择合适的超参数是模型调优的关键步骤之一。然而&#xff0c;由于超参数的组合空间通常非常庞大&#xff0c;手动调整超参数往往是一项耗时且困难的任务。为了解决这个问题&#xff0c;…

算法入门<二>:分治算法之汉诺塔问题及递归造成的栈溢出

1、分治算法 分治&#xff08;divide and conquer&#xff09;&#xff0c;全称分而治之&#xff0c;是一种非常重要且常见的算法策略。分治通常基于递归实现&#xff0c;包括“分”和“治”两个步骤。 分&#xff08;划分阶段&#xff09;&#xff1a;递归地将原问题分解为两…

PyCharm 2024新版图文安装教程(python环境搭建+PyCharm安装+运行测试+汉化+背景图设置)

名人说&#xff1a;一点浩然气&#xff0c;千里快哉风。—— 苏轼《水调歌头》 创作者&#xff1a;Code_流苏(CSDN) 目录 一、Python环境搭建二、PyCharm下载及安装三、解释器配置及项目测试四、PyCharm汉化五、背景图设置 很高兴你打开了这篇博客&#xff0c;如有疑问&#x…

小浪助手:下载学浪视频的最佳助手

小浪助手我已经打包好了,有需要的自己下载一下 学浪下载器链接&#xff1a;百度网盘 请输入提取码 提取码&#xff1a;1234 --来自百度网盘超级会员V10的分享 1.首先解压好我给大家准备好的压缩包 2.打开小浪助手.exe 3.选择一种登录方式&#xff0c;扫码登录或者手机号…

【办公类-26-02】20240423 UIBOT学分自动评价(自动登录、评价和退出,全自动)

背景需求&#xff1a; 我想用UIBOT自动模拟鼠标&#xff0c;登录每位老师的账户&#xff0c;进入评价区域&#xff0c;自动选择7次“满意”&#xff0c;输入1次“无”&#xff0c;然后提交。 C Dim objExcelWorkBook,arrayRet,iRet,temp,iPID,hWeb,dictRet,XobjExcelWorkBook …

警惕虚假宣传:GPT-4.0免费领取真相揭秘

警惕虚假宣传&#xff1a;GPT-4.0免费领取真相揭秘 在人工智能技术飞速发展的今天&#xff0c;尤其是OpenAI推出的GPT-4.0成为技术前沿的焦点&#xff0c;不少不法分子也开始借机进行欺诈。网络上出现了大量声称“免费领取GPT-4.0”的虚假信息&#xff0c;这不仅误导了公众&am…