【C++初阶】模拟实现优先级队列priority_queue

news2024/11/18 6:21:11

在这里插入图片描述

👦个人主页:@Weraphael
✍🏻作者简介:目前学习C++和算法
✈️专栏:C++航路
🐋 希望大家多多支持,咱一起进步!😁
如果文章对你有帮助的话
欢迎 评论💬 点赞👍🏻 收藏 📂 加关注✨


目录

  • 一、priority_queue的介绍
  • 二、为什么priority_queue不像stack和queue一样使用deque作为其底层存储数据的容器呢
  • 三、priority_queue的常见操作
  • 四、模拟实现priority_queue
      • 4.1 构造函数
      • 4.2 删除堆顶元素pop
      • 4.3 插入push
      • 4.4 获取堆顶元素top
      • 4.5 判断是否为空empty
      • 4.6 获取个数大小size
  • 五、仿函数
      • 5.1 什么是仿函数
      • 5.2 实现priority_queue的仿函数

一、priority_queue的介绍

在这里插入图片描述

  • priority_queue是一个容器适配器,默认使用vector作为其底层存储数据的容器
  • priority_queuevector上使用了堆heap的算法将vector中元素构造堆的结构,因此,priority_queue就是堆默认情况下是大堆。(如何构造成小堆在【仿函数】会讲解到)

二、为什么priority_queue不像stack和queue一样使用deque作为其底层存储数据的容器呢

  1. 这是因为vector在插入和删除元素时具有较好的性能表现。

在堆中插入新元素时,为了保持堆的特性(大堆/小堆)。则需要通过不断地比较和移动元素来完成。vector作为一个连续的内存块,可以更高效地进行元素的插入操作。

相比之下,deque在插入和删除操作时,需要考虑在数组的两端进行操作(头部和尾部),并且要进行元素的移动操作,使得时间复杂度稍高于vector

尽管deque在头部和尾部插入/删除操作上有一些优势,但对于优先级队列这种需要频繁访问和处理最高优先级元素的场景来说,vector更加合适

  1. 弹出pop元素:若要得到堆顶的元素,需要将堆顶元素与最后一个元素交换,并重新调整堆使其满足堆的性质。同样,由于vector是连续存储的,这个操作也可以更高效地进行。

  2. deque的存储空间不是连续的,因此在使用时需要更多的空间,可能会导致空间的浪费。

三、priority_queue的常见操作

在这里插入图片描述

#include <iostream>
#include <queue>
using namespace std;

// priority_queue的常见操作

int main()
{
	// 默认是大堆
	priority_queue<int> pq;
	pq.push(3);
	pq.push(5);
	pq.push(1);
	pq.push(4);
	pq.push(0);

	while (!pq.empty())
	{
		cout << pq.top() << ' ';
		pq.pop();
	}
	cout << endl;
	return 0;
}

【输出结果】

在这里插入图片描述

注意:优先级队列也是不支持迭代器遍历的!!!

四、模拟实现priority_queue

4.1 构造函数

namespace wj
{
	template<class T, class container = vector<T>>
	class priority_queue
	{	
	private:
		void AdjustDown(int parent)
		{
			// 建大堆
			// 找左右孩子大的
			size_t child = parent * 2 + 1;
			while (child < _con.size())
			{
				// 保证右孩子存在
				if (child + 1 < _con.size() && _con[child + 1] > _con[child]) 
				{
					++child;
				}

				if (_con[child] > _con[parent])
				{
					swap(_con[child], _con[parent]);
					parent = child;
					child = parent * 2 + 1;
				}
				else
				{
					break;
				}
			}
		}
		
	public:
		// 无参默认构造
		priority_queue()
		{}

		// 带区间的构造
		template<class InputIterator>
		priority_queue(InputIterator first, InputIterator last)
		{
			// 插入数据
			while (first != last)
			{
				_con.push_back(*first);
				++first;
			}
			//  建堆操作 (默认是大堆) 
			for (int i = (_con.size() - 1 - 1) / 2; i >= 0; i--)
			{
				AdjustDown(i);
			}
		}
	private:
		container _con;
	};
}

插入一个数据,由于还要保持大堆的性质,如果尾插的结点要比其父结点大,就要进行 向上调整

参考博客:点击跳转

4.2 删除堆顶元素pop

void pop()
{
	// 头尾结点交换,删除
	swap(_con[0], _con[_con.size() - 1]);
	_con.pop_back();
	// 然后再建堆
	AdjustDown(0);
}

4.3 插入push

void push(const T& val)
{
	// 尾部插入一个
	_con.push_back(val);
	// 再建堆
	AdjustUp(_con.size() - 1);
}

4.4 获取堆顶元素top

const T& top()
{
	return _con[0];
}

4.5 判断是否为空empty

bool empty()
{
	return _con.empty();
}

4.6 获取个数大小size

size_t size()
{
	return  _con.size();
}

五、仿函数

5.1 什么是仿函数

  • 仿函数(函数对象)是一种能够被像函数一样调用的对象。它通常是一个类或者结构体,重载了函数调用运算符 operator(),通过重载这个运算符,我们可以将对象当作函数来使用,实现了类似函数的行为。

  • 仿函数可以有自己的状态和成员变量,因此可以在多次调用中保持状态。它可以接受参数,并返回一个值。

例如,定义一个加法仿函数可以这样实现:

struct Add 
{
    int operator()(const int a, const int b) const 
    {
        return a + b;
    }
};

int main() 
{
    Add add;
    int result = add(3, 5);  // 调用仿函数
    cout << "add(3, 5) = " << result << endl;
    return 0;
}

在上面的例子中,Add是一个仿函数,它重载了函数调用运算符 operator(),使得add对象可以像函数一样被调用。通过add(3, 5),我们可以得到结果8

  • 在C++ 中,仿函数是一种灵活且强大的编程工具,常常用于算法和标准库中的函数对象。

在这里插入图片描述

就比如说sort函数,默认排的是升序

#include <iostream>
#include <algorithm>
using namespace std;

int main()
{
	int arr[] = { 5,1,4,2,0,3 };
	int arrSize = sizeof(arr) / sizeof(arr[0]);

	// less<int> 默认可以不写
	sort(arr, arr + arrSize, less<int>());

	for (int i = 0; i < arrSize; i++)
	{
		cout << arr[i] << ' ';
	}
	cout << endl;
	
	return 0;
}

【输出结果】

在这里插入图片描述

less是库里提供的,其作用就是用于小于不等式比较的函数对象类

在这里插入图片描述

那么如果想排降序,可以将less替换成greater,这也是库里提供的,其作用是用于大于不等式比较的函数对象类

在这里插入图片描述

#include <iostream>
#include <algorithm>
using namespace std;

int main()
{
	int arr[] = { 5,1,4,2,0,3 };
	int arrSize = sizeof(arr) / sizeof(arr[0]);

	// less<int> 默认可以不写
	sort(arr, arr + arrSize, greater<int>());

	for (int i = 0; i < arrSize; i++)
	{
		cout << arr[i] << ' ';
	}
	cout << endl;
	
	return 0;
}

【输出结果】

在这里插入图片描述

5.2 实现priority_queue的仿函数

namespace wj
{
	template<class T, class container = vector<T>, class Compare = less<T>>
	class priority_queue
	{
	private:
		void AdjustDown(int parent)
		{
			Compare com;
			// 建大堆
			// 找左右孩子大的
			size_t child = parent * 2 + 1;
			while (child < _con.size())
			{
				if (child + 1 < _con.size() && com(_con[child], _con[child + 1])) // 保证右孩子存在
				{
					++child;
				}

				if (com(_con[parent], _con[child]))
				{
					swap(_con[child], _con[parent]);
					parent = child;
					child = parent * 2 + 1;
				}
				else
				{
					break;
				}
			}
		}

		void AdjustUp(int child)
		{
			Compare com;

			int parent = (child - 1) / 2;
			while (child > 0)
			{
				if (com(_con[parent], _con[child]))
				{
					swap(_con[child], _con[parent]);
					child = parent;
					parent = (child - 1) / 2;
				}
				else
				{
					break;
				}
			}
		}
	public:
		// 无参默认构造
		priority_queue()
		{}

		// 带区间的构造
		template<class InputIterator>
		priority_queue(InputIterator first, InputIterator last)
		{
			// 插入数据
			while (first != last)
			{
				_con.push_back(*first);
				++first;
			}
			//  建堆操作 (默认是大堆) 
			for (int i = (_con.size() - 1 - 1) / 2; i >= 0; i--)
			{
				AdjustDown(i);
			}
		}

		void pop()
		{
			// 头尾结点交换,删除
			swap(_con[0], _con[_con.size() - 1]);
			_con.pop_back();
			// 然后再建堆
			AdjustDown(0);
		}

		void push(const T& val)
		{
			// 尾部插入一个
			_con.push_back(val);
			// 再建堆
			AdjustUp(_con.size() - 1);
		}

		const T& top()
		{
			return _con[0];
		}

		bool empty()
		{
			return _con.empty();
		}

		size_t size()
		{
			return  _con.size();
		}

	private:
		container _con;
	};
}

注意:如果在priority_queue中放自定义类型的数据,用户需要在自定义类型中提供>或者<的重载。

namespace wj
{
	template<class T, class container = vector<T>, class Compare = less<T>>
	class priority_queue
	{
	private:
		void AdjustDown(int parent)
		{
			Compare com;
			// 建大堆
			// 找左右孩子大的
			size_t child = parent * 2 + 1;
			while (child < _con.size())
			{
				// 改动
				if (child + 1 < _con.size() && com(_con[child], _con[child + 1])) // 保证右孩子存在
				{
					++child;
				}

				if (com(_con[parent], _con[child]))
				{
					swap(_con[child], _con[parent]);
					parent = child;
					child = parent * 2 + 1;
				}
				else
				{
					break;
				}
			}
		}

		void AdjustUp(int child)
		{
			Compare com;

			int parent = (child - 1) / 2;
			while (child > 0)
			{
				if (com(_con[parent], _con[child]))
				{
					swap(_con[child], _con[parent]);
					child = parent;
					parent = (child - 1) / 2;
				}
				else
				{
					break;
				}
			}
		}
	public:
		// 无参默认构造
		priority_queue()
		{}

		// 带区间的构造
		template<class InputIterator>
		priority_queue(InputIterator first, InputIterator last)
		{
			// 插入数据
			while (first != last)
			{
				_con.push_back(*first);
				++first;
			}
			//  建堆操作 (默认是大堆) 
			for (int i = (_con.size() - 1 - 1) / 2; i >= 0; i--)
			{
				AdjustDown(i);
			}
		}

		void pop()
		{
			// 头尾结点交换,删除
			swap(_con[0], _con[_con.size() - 1]);
			_con.pop_back();
			// 然后再建堆
			AdjustDown(0);
		}

		void push(const T& val)
		{
			// 尾部插入一个
			_con.push_back(val);
			// 再建堆
			AdjustUp(_con.size() - 1);
		}

		const T& top()
		{
			return _con[0];
		}

		bool empty()
		{
			return _con.empty();
		}

		size_t size()
		{
			return  _con.size();
		}

	private:
		container _con;
	};
	
	// 日期类
	class Date
	{
	public:
		Date(int year = 1900, int month = 1, int day = 1)
			: _year(year)
			, _month(month)
			, _day(day)
		{}
		bool operator<(const Date& d)const
		{
			return (_year < d._year) ||
				(_year == d._year && _month < d._month) ||
				(_year == d._year && _month == d._month && _day < d._day);
		}
		bool operator>(const Date& d)const
		{
			return (_year > d._year) ||
				(_year == d._year && _month > d._month) ||
				(_year == d._year && _month == d._month && _day > d._day);
		}
		friend ostream& operator<<(ostream& _cout, const Date& d);
	private:
		int _year;
		int _month;
		int _day;
	};

	ostream& operator<<(ostream& _cout, const Date& d)
	{
		_cout << d._year << "-" << d._month << "-" << d._day;
		return _cout;
	}
}

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

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

相关文章

Elasticsearch:将段落向量搜索添加到 Lucene

作者&#xff1a;Benjamin Trent 向量搜索是信息检索工具箱中的一个强大工具。 将向量与词法搜索&#xff08;如 BM25&#xff09;一起使用很快变得司空见惯。 但向量搜索中仍然存在一些痛点需要解决。 主要的一个是文本嵌入模型和处理更大的文本输入。 像 BM25 这样的词法搜索…

数据库备份与恢复

数据库备份的重要性 在生产环境中&#xff0c;数据的安全性至关重要&#xff0c;任何数据的丢失都可能产生严重的后果。 造成数据丢失的原因有&#xff1a;程序错误、人为操作错误、运算错误、磁盘故障、灾难(如火灾、地震)和盗窃。 数据库备份的分类 从物理与逻辑的角度&a…

【多面体:知识蒸馏:Pansharpening】

Multipatch Progressive Pansharpening With Knowledge Distillation &#xff08;基于知识蒸馏的多面体渐进锐化算法&#xff09; 在这篇文章中&#xff0c;我们提出了一种新的多面体和多级泛锐化方法与知识蒸馏&#xff0c;称为PSDNet。不同于现有的pansharpening方法&…

ASP.NET Core 中基于 Controller 的 Web API

基于 Controller 的 Web API ASP.NET Wep API 的请求架构 客户端发送Http请求&#xff0c;Contoller响应请求&#xff0c;并从数据库读取数据&#xff0c;序列化数据&#xff0c;然后通过 Http Response返回序列化的数据。 ControllerBase 类 Web API 的所有controllers 一般…

中央空调秒变智能 青岛中弘P15面板式空调网关初体验

在智能家居逐步渗透进千家万户的今天&#xff0c;如何将中央空调融入到智能化场景&#xff0c;以实现场景联动、提升家居生活的智能化和科技化程度&#xff0c;中弘给出了新的答案。本期智哪儿就带大家测评一下青岛中弘P15面板式空调网关&#xff0c;一起看看它的价值所在。 高…

The Sandbox 即将参加韩国区块链周,并带来一系列独家周边活动!

韩国区块链周&#xff08;Korea Blockchain Week&#xff09;即将到来&#xff0c;届时将有成千上万的 NFT 项目、建设者、社区成员、企业家、投资者和爱好者齐聚首尔&#xff0c;分享 Web3 的最新更新和未来愿景。 继成功举办韩流崛起 LAND 销售并宣布多个合作伙伴关系之后&a…

计算机毕设之基于python+echarts+mysql的图书馆可视化管理系统(文档+代码+部署教程)

系统阐述的是一款图书馆可视化管理系统的设计与实现&#xff0c;对于Python、B/S结构、MySql进行了较为深入的学习与应用。主要针对系统的设计&#xff0c;描述&#xff0c;实现和分析与测试方面来表明开发的过程。开发中使用了 django框架和MySql数据库技术搭建系统的整体架构…

群晖NAS:DSM7.1激活Advanced Media Extensions【自留记录】

群晖NAS&#xff1a;DSM7.1激活Advanced Media Extensions【自留记录】 本文仅半白群晖可用&#xff0c;不需要安装其他套件或者ssh修改什么 使用DS Video 网页播放视频时候&#xff0c;提示&#xff1a;【不支持当前所选音轨的文件格式&#xff0c; 因此无法播放视频。请尝试…

编程题四大算法思想(二)——回溯法:N皇后问题、子集和问题、地图填色问题、迷宫问题

文章目录 回溯法迷宫游戏 N皇后问题基本概念解空间4后问题的解空间 可行解和最优解回溯法回溯法术语回溯法的关键问题回溯法的基本思想4后问题的约束条件n后问题生成问题状态的基本方法 子集和问题一个朴素的求解方法回溯回溯法的剪枝技术 地图填色问题 回溯法 迷宫游戏 深度优…

Android RecyclerView 之 吸顶效果

前言 上一篇文章已经实现了列表跟宫格布局的动态切换&#xff0c;这篇文章主要来说通过 CoordinatorLayout 和 AppbarLayout 的配合&#xff0c;以及 NestedScrollView 来实现吸顶效果 。效果如下。 一、CoordinatorLayout 是什么&#xff1f; CoordinatorLayout 是 Androi…

pnpm快速创建 Vue.js 项目(npm类似)

目录 pnpm 创建一个 Vue.js 项目 前提准备&#xff1a; 运行创建命令&#xff1a; 选择项目配置&#xff1a;&#xff08;按需选择&#xff09; cd 项目名&#xff1a;&#xff08;进入项目终端&#xff09; 安装项目依赖&#xff1a; 运行项目&#xff1a; pnpm 创建一…

文献速读|5分的生信+免疫组化:单细胞测序转录组联合bulk转录组肿瘤预后模型

今天给大家分享一篇IF5.8的纯生信单细胞联合Bulk转录组构建预后模型的文章&#xff0c;于2023年3月19日发表在Cancer Immunology Immunotherapy上&#xff1a;Integrative analyses of bulk and single-cell RNA-seq identified cancer-associated fibroblasts-related signatu…

Docker安装详细步骤

Docker安装详细步骤 1、安装环境准备 主机&#xff1a;192.168.40.5 zch01 设置主机名 # hostnamectl set-hostname zch01 && bash 配置hosts文件 [root ~]# vi /etc/hosts 添加如下内容&#xff1a; 192.168.40.5 zch01 关闭防火墙 [rootzch01 ~]# systemct…

W5500-EVB-PICO通过SNTP获取网络时间(十一)

前言 上一章我们用W5500_EVB_PICO 开发板做Ping数据测试IP检测连通性&#xff0c;那么本章我们进行W5500_EVB_PICO SNTP的测试。 什么是NTP&#xff1f; NTP(Network Time Protocol&#xff09;网络时间协议基于UDP&#xff0c;用于网络时间同步的协议&#xff0c;使网络中的计…

如何高效进行测试用例评审

1.用例评审的目的 为了减少测试人员执行阶段做无效工作&#xff0c;执行无效case&#xff0c;提交无效缺陷&#xff08;可以友情提醒研发同学&#xff0c;讲到自己负责的相关模块时&#xff0c;注意下是否存在异议点&#xff09;为了避免三方&#xff08;产品、研发、测试&…

多项式求逆

已知 F F F&#xff0c;求 G G G 考虑倍增 F ( x ) ∗ H ( x ) ≡ 1 ( m o d x n / 2 ) F(x) * H(x) \equiv 1 \pmod{x^{n/2}} F(x)∗H(x)≡1(modxn/2) F ( x ) ∗ G ( x ) ≡ 1 ( m o d x n / 2 ) F(x) * G(x) \equiv 1 \pmod{x^{n/2}} F(x)∗G(x)≡1(modxn/2) 假设 H H…

噪声的产生机理和来源

引言&#xff1a;噪声广泛存在于自然界&#xff0c;上节揭示了噪声的本质&#xff0c;噪声按照噪声携带能量的强弱分为功率型噪声和信号型噪声&#xff0c;功率型噪声持续时间短&#xff0c;能量强&#xff0c;对设备的寿命具有很大的影响&#xff0c;而信号型噪声顾名思义来源…

不想出门?那就把“自然”搬进家里吧!

为自然主题房间寻找灵感&#xff1a; 简单创意 以自然为主题的房间将自然灵感的色调&#xff08;棕色、灰色、米色、白色、蓝色和绿色&#xff09;与皮革、木材、黄麻和藤条等纹理相结合。就像真实的户外一样&#xff0c;它也懂得平衡的力量。户外&#xff1a; 每一片树丛&…

云备份——第三方库简单介绍并使用(上)

目录 一&#xff0c;Jsoncpp库序列化和反序列化 二&#xff0c;bundle文件压缩库 2.1 文件压缩 2.2 文件解压 一&#xff0c;Jsoncpp库序列化和反序列化 首先我们需要先了解一下json是什么&#xff0c;json是一种数据交换格式&#xff0c;采用完全独立于编程语言的文本格式来…

开学有哪些电容笔值得买?平价电容笔排行榜

苹果的原装Pencil&#xff0c;无疑是一款性能出色的电容笔&#xff0c;但它的价格也很高&#xff0c;如果不小心弄丢了或者弄坏了&#xff0c;那就太让人心痛。再说了&#xff0c;一支价值不菲的电容笔&#xff0c;要是不是用于专业绘画&#xff0c;实在是大材小用。不过&#…