首页 > 编程语言 >【C++】priority_queue的模拟实现与仿函数

【C++】priority_queue的模拟实现与仿函数

时间:2024-06-21 18:57:16浏览次数:25  
标签:priority 容器 year C++ queue child size con

文章目录

在这里插入图片描述

1.优先级队列的介绍与使用

1.1 介绍

  1. 优先级队列是一种容器适配器,根据严格的弱排序标准,它的第一个元素总是它所包含的元素中最大的

  2. 其内容似于堆,在堆中可以随时插入元素,并且只能检索最大/小堆元素(优先队列中位于顶部的元素)。

  3. 优先队列被实现为容器适配器,容器适配器即将特定容器类封装作为其底层容器类,queue提供一组特定的成员函数来访问其元素。元素从特定容器的“尾部”弹出,其称为优先队列的顶部。

  4. 底层容器可以是任何标准容器类模板,也可以是其他特定设计的容器类。容器应该可以通过随机访问迭代器访问,并支持以下操作:

    • empty():检测容器是否为空
    • size():返回容器中有效元素个数
    • front():返回容器中第一个元素的引用
    • push_back():在容器尾部插入元素
    • pop_back():删除容器尾部元素
  5. 标准容器类vector和deque满足这些需求。默认情况下,如果没有为特定的priority_queue类实例化指定容器类,则使用vector。

  6. 需要支持随机访问迭代器,以便始终在内部保持堆结构。容器适配器通过在需要时自动调用算法函数
    make_heap、push_heap和pop_heap来自动完成此操作。

1.2 使用

优先级队列默认使用vector作为其底层存储数据的容器,在vector上又使用了堆算法将vector中元素构造成堆的结构,因此priority_queue就是堆,所有需要用到堆的位置,都可以考虑使用priority_queue
注意:默认情况下priority_queue是大堆

在这里插入图片描述

在这里插入图片描述

如果想让它是小堆应该怎么办呢?
设置它的模板参数(这里是仿函数,后面介绍)

在这里插入图片描述

2. 模拟实现

我们priority_queue的实现仍然使用适配器的模式,由于它的逻辑结构是堆结构,所以我们默认使用vector作为默认容器。我们只需要在内部实现堆的算法就可以了。

namespace qp
{
	//<类型,适配器> 
	template<class T ,class Continer = vector<T>>
	class priority_queue
	{
	public:

	private:
		Continer _con;
	};
}

2.1 push

这里需要使用堆的向上调整算法。

		void AdjustUp(size_t child)
		{
			size_t parent = (child - 1) / 2;
			while (child > 0)
			{
				//大堆
				if (_con[parent] < _con[child])
				{
					swap(_con[parent], _con[child]);
					child = parent;
					parrent = (child - 1) / 2;
				}
				else
				{
					break;
				}
			}
		}

		//先将数据插入到尾部,为了保持堆的状态,需要向上调整
		void push(const T& x)
		{
			_con.push_back(x);
			AdjustUp(_con.size() - 1);
		}

在这里插入图片描述

2.2 pop

这里需要使用堆的向下调整算法。

		void AdjustDown(size_t parent)
		{
			size_t child = 2 * parent + 1;
			//若孩子存在
			while (child < _con.size())
			{
				if (child + 1 < _con.size() && _con[child] < _con[child + 1])
				{
					child++;
				}

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

		void pop()
		{
			//先将堆顶与尾部换
			swap(_con[0], _con[_con.size() - 1]);
			//尾部不看做堆里的
			_con.pop_back();
			//为了保持堆的状态,需要向下调整
			AdjustDown(0);
		}

在这里插入图片描述

2.3 top、empty、size

		T& top()
		{
			return _con[0];
		}
		
		size_t size()
		{
			return _con.size();
		}

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

在这里插入图片描述

2.4 迭代区间构造

		priority_queue() = default;//使用编译器生成的默认构造

		template<class InputIterator>
		priority_queue(InputIterator first, InputIterator last)
		{
			while (first != last)
			{

				//push(*first); //若直接push,使用的是向上调整建堆,时间复杂度为:(N*logN)
				_con.push_back(*first);
				++first;
			}

			//从第一个非叶子开始,向下调整建堆,时间复杂度为:(N)
			for (int i = (_con.size() - 1 - 1) / 2; i >= 0; i--)
			{
				AdjustDown(i);
			}
		}

3.仿函数

  1. 什么是仿函数?

仿函数/函数对象:重载了operator()的类,类的对象可以像函数一样使用

在这里插入图片描述

operator() 的特点:参数个数和返回值根据需求确定,不固定,很灵活

而且仿函数是一个类,那么类就可以写成模板,我们可以通过实例化来控制它。

在这里插入图片描述

所以仿函数到底有什么用呢?
我们上面模拟实现的priority_queue默认是大堆,那如果我想要一个小堆呢?

在这里插入图片描述

我们也发现官方文档中也是有三个模板参数的。

在这里插入图片描述
所以我们怎么写呢?

在这里插入图片描述
既然仿函数是一个类,那么我们就可以让它作为模板参数,默认是使用小于。

仿函数的类可以像对象一样去使用,所以我们可以这样用:

在这里插入图片描述
此时,我们在外部就可以控制它是大堆还是小堆了。

此处也可将cmp定义为类的成员变量

在这里插入图片描述

测试

在这里插入图片描述

尽管C语言使用函数指针也可以实现,但是C++更倾向于使用仿函数。虽然写法上麻烦了,但是使用更加灵活了。

  1. 对于自定义类型

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

下面的代码中,Date类提供了比较相关的重载

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;
}

此时是完全没有问题的

在这里插入图片描述

如果我们让它存一个Date*呢?

在这里插入图片描述

输出的结果是不固定的,它拿每个Date的地址去比较了,与我们所期望得到的是不一致的。所以它的比较那块就出现了问题,我们就可以自己去写一个比较功能了。

此时的结果就是正确的。

在这里插入图片描述

所以,仿函数除了控制大堆、小堆,还能控制比较逻辑,如果默认的比较逻辑不是你想要的,或它不支持比较大小,那我们均可以通过仿函数控制
在这里插入图片描述

标签:priority,容器,year,C++,queue,child,size,con
From: https://blog.csdn.net/weixin_69380220/article/details/139861792

相关文章

  • 校招常见七大排序C++版(适合新人,通俗易懂)
    作者:求一个demo版权声明:著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处内容通俗易懂,没有废话,文章最后是面试常问内容是否稳定最优、最坏、平均时间复杂度最优、最坏、平均空间复杂度冒泡排序是O(n)、O(n^2)、O(n^2)0、O(n)、O(1)选择排序否O(n^2)、O(n^2)......
  • C++矩阵库:Eigen 3.4.90 中文使用文档 (一)
    写在前面:我在学习Eigen库时,没找到好的中文文档,因此萌发了汉化Eigen官网文档的想法。其中一些翻译可能不是特别准确,欢迎批评指正。感兴趣的同学可以跳转到官网查看原文:Eigen:MainPagehttps://eigen.tuxfamily.org/dox/index.html       Eigen库,是一个开源的C......
  • C/C++ 缓冲区溢出问题总结
    缓冲区溢出(BufferOverflow)是一种常见的安全漏洞,它发生在当程序试图将更多的数据放入一个固定大小的内存区域(即缓冲区)时,超过了该区域所能容纳的数据量。这可能导致未定义的行为,包括数据损坏、程序崩溃,甚至更糟糕的是,攻击者可以利用这种漏洞执行恶意代码。一、缓冲区溢出概述缓冲......
  • C++ 面向对象高级开发 4、参数传递与返回值
    consructor构造函数:被放在private区ctors放在private区classA{public:staticA&getInsance();    setup(){...};private:A();    A(constA&rhs);};A&A::getInstance(){staticAa;    returna;}A::getInsance().s......
  • [Effective Modern C++] 条款18笔记
    条款18中的完整代码:点击查看代码#include<iostream>#include<memory>#include<string>//假设基础的Investment类和Stock类classInvestment{public:virtual~Investment()=default;virtualvoiddisplay()const=0;};//其它类类似,略classSto......
  • [Effective Modern C++] 条款19笔记 - 为什么deleter的类型是std::unique_ptr类型的一
    为什么deleter的类型是std::unique_ptr类型的一部分,而不是std::shared_ptr的一部分?std::unique_ptr<Widget,decltype(loggingDel)>upw(newWidget,loggingDel);std::shared_ptr<Widget>upw(newWidget,loggingDel);这个问题涉及到std::unique_ptr和std::shared_ptr......
  • C++list类的常见函数以及其模拟实现
    文章目录前言一、list内部成员有什么?二、构造函数以及析构函数1.默认构造2.传参构造3.迭代器构造4.深拷贝以及运算符=的重载1.深拷贝2.=的重载5.析构函数三、迭代器的模拟实现1.正向迭代器2.反向迭代器四、常见函数及其实现1.insert函数2.erase函数3.clear函数4.push_......
  • 我一直看不明白:“C++会被java/python等这些语言替代”
    在开始前刚好我有一些资料,是我根据网友给的问题精心整理了一份「C++的资料从专业入门到高级教程」,点个关注在评论区回复“888”之后私信回复“888”,全部无偿共享给大家!!!有些程序,是既可以用c++编写,也可以用java/python编写。如果这类程序以前主要是由c++编写,后来逐渐变成主要......
  • c++重载输出流(<<)
    一.重载输出流在C++中,可以重载输出流运算符(<<)来自定义对象的输出方式。重载输出流运算符允许我们以自定义的方式将对象输出到标准输出流或其他输出流中。以下是关于重载输出流运算符(<<)的几个知识点以及相应的示例:重载输出流运算符的语法:重载输出流运算符必须作为一个普......
  • 《C++ Primer》导学系列:第 7 章 - 类
    7.1定义抽象数据类型7.1.1类的基本概念在C++中,类是用户定义的类型,提供了一种将数据和操作这些数据的函数(成员函数)组合在一起的方法。类定义了对象的属性和行为,通过实例化类来创建对象。7.1.2定义类定义类时,需要指定类的名称,并在类体内声明数据成员和成员函数。类定义的......