当前位置: 首页 > article >正文

【STL】priority_queue的使用和模拟实现

priority_queue的使用和介绍

priority_queue的介绍

优先级队列(priority_queue)它也是一个容器适配器,默认使用的容器是vector,在往队列中放入数据时,它会将容器中放置的数据通过堆算法调整成 大堆/小堆,在使用时我们若不指定建堆方法,它默认会是排大堆(less)

它的使用方法其实和stack,queue类似,只是priority_queue在插入元素的基础上进行了调整建堆

priority_queue的定义方式

1.使用priority_queue创建大堆对象:

priority_queue<int, vector<int>, less<int>> q1;

2.使用priority_queue创建小堆对象:

priority_queue<int ,vector<int>, less<int>> q2;

3.不指定容器和底层使用的堆算法

priority_queue<int> q3;

这时它的容器和堆算法默认使用的是vector<int>和less<int>

priority_queue的接口展示:

成员函数功能
push向队尾插入一个元素,并向上调整排序
pop删除堆顶元素(首尾元素交换,删除队尾元素,首元素向下调整)
top访问堆顶元素(队头元素)
size获取队列中有效元素个数
 empty判断队列是否为空
swap交换两个队列的元素

priority_queue的模拟实现

模拟实现:

//仿函数/函数对象
//()运算符重载, 使用方式类似于函数调用,所以可以称为:仿函数
//建大堆
template<class T>
struct Less
{
	bool operator()(const T& x, const T& y)
	{
		return x > y;
	}
};

//建小堆
template<class T>
struct Greater
{
   bool operator()(const T& x, const T& y)
   {
        return x < y;
   }
};

template<class T, class container = vector<T>, class compare = Less<T>>
class priority_queue
{
	//向上调整, 排大堆
	void AdjustUp(size_t child)
	{
		size_t parent = (child - 1) / 2;

		while (child > 0)
		{
			if (_com(_con[child], _con[parent])) //使用所给比较方式进行比较
			{
				swap(_con[child], _con[parent]);
				child = parent;
				parent = (child - 1) / 2;
			}
			else
			{
				break;
			}
		}
	}

	//向下调整,排大堆
	void AdjustDown(size_t parent)
	{
		compare con; 

		size_t child = parent * 2 + 1;

		while (child < _con.size())
		{
			if (child + 1 < _con.size() && _com(_con[child + 1], _con[child]))
			{
				++child;
			}

			if (_com(_con[child], _con[parent]))
			{
				swap(_con[parent], _con[child]);
				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 (size_t end = (_con.size() - 2) / 2; end >= 0; ++end)
		{
			AdjustDown(end);
		}
	}

	void push(const T& key)
	{
		_con.push_back(key);
		AdjustUp(_con.size() - 1);
	}
	
	void pop()
	{
		swap(_con[0], _con[_con.size() - 1]);
		_con.pop_back();
		AdjustDown(0);
	}

	T& top()
	{
		return _con.front();
	}

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

	bool empty()
	{
		return _con.empty();
	}
private:
	container _con;
    compare _com;
};


http://www.kler.cn/a/389221.html

相关文章:

  • C# 网络协议第三方库Protobuf的使用
  • Kotlin基础知识学习(三)
  • 5G网络下移动机器人的图像和指令传输用于远程操作
  • Flutter:自定义Tab切换,订单列表页tab,tab吸顶
  • Open3D计算点云粗糙度(方法一)【2025最新版】
  • 一个软件分发和下载的网站源码,带多套模板
  • 无网络安装ionic和运行
  • 【专题】事务与并发控制
  • 计算机视觉基础:OpenCV库详解
  • 【后端速成Vue】computed计算属性
  • Ardusub中添加自定义控制器
  • 计算机网络-1.2分层结构
  • SQLite 与 Python:集成与使用
  • 七次课掌握 Photoshop:选区与抠图
  • ORACLE创建用户之后查询不到创建的用户
  • React 守卫路由
  • 测试用例设计方法之边界值分析法
  • Dependencies 工具
  • node 阿里云oss上传删除修改文件
  • vue3的自定义hooks怎么写?
  • 《深入浅出Apache Spark》系列③:Spark SQL解析层优化策略与案例解析
  • Redis的缓存问题与应对策略
  • 面试:TCP、UDP如何解决丢包问题
  • 探索开放资源上指令微调语言模型的现状
  • 【Kafka-go】golang的kafka应用
  • ReactPress:深入解析技术方案设计与源码