【C++/STL】:优先级队列的使用及底层剖析仿函数

目录

  • 💡前言
  • 一,优先级队列的使用
  • 二,仿函数
    • 1,什么是仿函数
    • 2,仿函数的简单示例
  • 三,优先级队列的底层剖析

💡前言

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

注意:使用优先级队列要包含头文件 < queue >

一,优先级队列的使用

在这里插入图片描述

代码实现如下:

这里的建堆一般有两种方式:
(1) 一种是一个一个push进vector容器再进行向上调整建堆
(2) 另一种是直接用迭代器区间构造直接建堆(推荐用这种)

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

void test_priority_queue()
{
	vector<int> v = { 6,0,3,5,4,7,9,1,2,8 };

	//默认升序
	//priority_queue<int> pq(v.begin(), v.end());

	//一个一个尾插建堆
	priority_queue<int, vector<int>, greater<int>> pq;
	for (auto e : v)
	{
		pq.push(e);
	}

	//迭代器区间构造,直接建堆
	//priority_queue<int,vector<int>,greater<int>> pq(v.begin(), v.end());

	while (!pq.empty())
	{
		cout << pq.top() << " ";
		pq.pop();
	}
	cout << endl;

}

int main()
{
	test_priority_queue();

	return 0;
}

注意:优先级队列默认的大堆,降序排列,如果要升序,就要换仿函数。下图中第三个模板参数就是传仿函数。

使用算法库里的 less 和 greater 算法,需要包含头文件< functional >

在这里插入图片描述

二,仿函数

1,什么是仿函数

仿函数也叫函数对象,是一个重载了 operator() 的类,可以使得类的对象像函数一样使用

2,仿函数的简单示例

operator()并没有参数的个数和返回值,所以使用是十分灵活的

struct Func1
{
	//无参无返回值
	void operator()()
	{
		cout << "Func调用" << endl;
	}
};

struct Func2
{
	//有参无返回值
	void operator()(int n)
	{
		while (n--)
		{
			cout << "Func调用" << endl;
		}
	}
};

int main()
{
	Func1 f1;
	f1();  //使得对象像函数一样使用
	f1.operator()(); //显示调用

	cout << endl;

	Func2 f2;
	f2(3);  //使得对象像函数一样使用

	return 0;
}

在这里插入图片描述

三,优先级队列的底层剖析

namespace ling
{
	template<class T>
	class myless
	{
	public:
	    bool operator()(const T& x, const T& y)
	    {
	        return x < y;
	    }
	};
	
	template<class T>
	class mygreater
	{
	public:
	    bool operator()(const T& x, const T& y)
	    {
	        return x > y;
	    }
	};
	
	template <class T, class Container = vector<T>, class Compare = myless<T>>
	class priority_queue
	{
	public:
	    priority_queue() = default;
		
		//迭代器区间构造
	    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--)
	        {
	            Adjust_down(i);
	        }
	    }
	
	    bool empty() const
	    {
	        return con.empty();
	    }
	
	    size_t size() const
	    {
	        return con.size();
	    }
		
		// 堆顶元素不允许修改,因为:堆顶元素修改可以会破坏堆的特性
	    const T& top()const
	    {
	        return con[0];
	    }
	
	    //向上调整
	    void Adjust_up(int child)
	    {
	        int parent = (child - 1) / 2;
	        while (child > 0)
	        {
	            //if (con[parent] < con[child])
	            if(comp(con[parent], con[child]))
	            {
	                swap(con[parent], con[child]);
	                child = parent;
	                parent = (child - 1) / 2;
	            }
	            else
	            {
	                break;
	            }
	        }
	    }
	
	    void push(const T& x)
	    {
	        con.push_back(x);
	        Adjust_up(con.size() - 1);
	    }
	
	    //向下调整
	    void Adjust_down(int parent)
	    {
	        int child = parent * 2 + 1;
	        while (child < con.size())
	        {
	            if (child + 1 < con.size() && comp(con[child], con[child + 1]))
	            {
	                child += 1;
	            }
	            //if (con[parent] < con[child])
	            if(comp(con[parent], con[child]))
	            {
	                swap(con[parent], con[child]);
	                parent = child;
	                child = parent * 2 + 1;
	            }
	            else
	            {
	                break;
	            }
	        }
	    }
	
	    void pop()
	    {
	        swap(con[0], con[con.size() - 1]);
	        con.pop_back();
	
	        Adjust_down(0);
	    }
	private:
	    Container con;
	    Compare comp;
	};
}

测试代码

void TestQueuePriority()
{
	ling::priority_queue<int> q1;
	q1.push(5);
	q1.push(1);
	q1.push(4);
	q1.push(2);
	q1.push(3);
	q1.push(6);
	cout << q1.top() << endl;

	q1.pop();
	q1.pop();
	cout << q1.top() << endl;

	vector<int> v{ 5,1,4,2,3,6 };
	ling::priority_queue<int, vector<int>, ling::greater<int>> q2(v.begin(), v.end());
	cout << q2.top() << endl;

	q2.pop();
	q2.pop();
	cout << q2.top() << endl;
}

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

改变AI历史的Transformer是如何帮助LLM大模型工作的?看图解密Transformer原理,看不懂算我输!

在过去的几年里&#xff0c;大型语言模型(LLM)的出现&#xff0c;为长达数十年的智能机器构建的探索中带来了巨大的飞跃。 这项基于试图模拟人类大脑的研究技术&#xff0c;也在近几年催生了一个新领域——Generative AI 生成式人工智能&#xff0c;简单理解就是可以 通过模仿…

4.制作的docker镜像

最近工作需要&#xff0c;制作docker镜像&#xff0c;用做构建使用。 1.拉取基础镜像ubuntu:22.04 docker pull ubuntu:22.042.运行ubuntu容器 docker run --privileged -d --name ubuntu_build ubuntu:22.04 sleep infinity3.进入运行的容器 docker exec -it ubuntu_build …

数据结构:队列详解 c++信息学奥赛基础知识讲解

目录 一、队列概念 二、队列容器 三、队列操作 四、代码实操 五、队列遍历 六、案例实操 题目描述&#xff1a; 输入格式&#xff1a; 输出格式&#xff1a; 输入样例&#xff1a; 输出样例&#xff1a; 详细代码&#xff1a; 一、队列概念 队列是一种特殊的线性…

激励视频广告的eCPM更高,每天的展示频次有限制吗?

在APP发展初期&#xff0c;由于DUA量级有限&#xff0c;所需的广告资源比较少&#xff0c;往往接入1-2家广告平台就能满足APP用户每日需要的广告展示量。而随着APP用户规模的扩大、广告场景的不断丰富&#xff0c;开发者要提升APP整体广告变现收益&#xff0c;一是可以尽可能多…

nacos使用shared-configs设置多个配置文件后,配置中修改无法动态更新 解决办法

问题描述 今天使用nacos去做配置分离&#xff0c;启动成功了&#xff0c;配置也读取了&#xff0c;但是当我修改nacos中的配置时&#xff0c;发现数据无法动态更新 下面是测试接口的调用 可以看到我修改配置后&#xff0c;接口返回的参数依然是老参数 问题排查 首先检查了…

重生奇迹MU新手攻略:如何一步步往大佬发展

装备强化攻略&#xff1a; 提纯装备&#xff1a;通过提纯装备可以提升基础属性&#xff0c;选择合适的装备进行提纯可以获得更好的效果。 镶嵌宝石&#xff1a;使用宝石进行装备镶嵌可以增加装备的属性&#xff0c;根据需要选择适合的宝石进行镶嵌。 洗练装备&#xff1a;通…

大模型赋能全链路可观测性:运维效能的革新之旅

目录 全链路可观测工程与大模型结合---提升运维效能 可观测性&#xff08;Observability&#xff09;在IT系统中的应用及其重要性 统一建设可观测数据 统一建设可观测数据的策略与流程 全链路的构成和监控形态 云上的全链路可视方案 为什么一定是Copilot 大模型的Copilo…

jenkins设置定时构建语法

一、设置定时 定时构建的语法是*** * * * ***。 第一个*表示分钟&#xff0c;取值范围是0~59。例如&#xff0c;5 * * * *表示每个小时的第5分钟会构建一次&#xff1b;H/15 * * * 或/15 * * * 表示每隔15分钟构建一次&#xff1b; 第2个表示小时&#xff0c;取值范围是0~23。…

气膜建筑审批流程及现状分析—轻空间

气膜建筑作为一种新兴的建筑形式&#xff0c;以其快速建造、成本低廉和灵活多变的优势在各个领域得到了广泛应用。然而&#xff0c;气膜建筑在我国尚未被纳入正式的建筑规范&#xff0c;这使得其审批流程与传统建筑有显著差异。轻空间将详细探讨气膜建筑的审批流程及其在实际操…

全局mixins

一、文章由来 在开发过程中发现在钩子函数位置直接使用dicts就能直接绑定数据了&#xff0c;由此溯源发现了自己的盲区 二、局部使用 // myMixin.js文件 var myMixin {created: function () {this.hello()},methods: {hello: function () {console.log(hello from mixin!)…

Transformers 安装与基本使用

文章目录 Github文档推荐文章简介安装官方示例中文情感分析模型分词器 Tokenizer填充 Padding截断 Truncation google-t5/t5-small使用脚本进行训练Pytorch 机器翻译数据集下载数据集格式转换 Github https://github.com/huggingface/transformers 文档 https://huggingface…

边缘计算VNC智能盒子如何助力HMI设备实现二次开发?

HMI&#xff08;Human-Machine Interface&#xff09;又称人机界面&#xff0c;是用户与机器之间交互和通信的媒介。今天带你了解智能盒子如何助力HMI设备实现二次开发&#xff1f; HMI设备被广泛应用在工业自动化中&#xff0c;具有显示设备信息&#xff0c;实时监测&#xf…

【Linux杂货铺】Linux学习之路:期末总结篇1

第一章 什么是Linux? Linux 是 UNIX 操作系统的一个克隆&#xff1b;它由林纳斯 本纳第克特 托瓦兹从零开始编写&#xff0c;并在网络上众多松散的黑客团队的帮助下得以发展和完善&#xff1b;它遵从可移植操作系统接口&#xff08;POSIX&#xff09;标准和单一 UNIX 规范…

短信群发策略优化:如何有效降低退订率?

在短信群发营销中&#xff0c;退订率的上升常常影响营销效果。为了降低退订率&#xff0c;提高客户黏性&#xff0c;以下是一些实用的策略建议&#xff1a; 1.合理控制发送频率 过多的短信发送会给客户带来骚扰感&#xff0c;导致退订。因此&#xff0c;应合理控制短信的发送频…

排序算法(C语言版)

前言 排序作为生产环境中常见的需求之一&#xff0c;对整个产品有举足轻重的影响&#xff0c;可以说使用一个合适的排序算法是业务逻辑中比较重要的一部分。今天我们就来介绍常见的排序算法以及实现 排序 所谓排序无非就是按照特定的规则对一组数据就行顺序化。 常见的排序有…

智能语音热水器:置入NRK3301离线语音识别ic 迈向智能家居新时代

一、热水器语音识别芯片开发背景 在科技的今天&#xff0c;人们对于生活品质的追求已不仅仅满足于基本的物质需求&#xff0c;更渴望通过智能技术让生活变得更加便捷、舒适。热水器作为家庭生活中不可或缺的一部分&#xff0c;其智能化转型势在必行。 在传统热水器使用中&#…

论文导读 | 事件因果关系抽取和识别

导读 目前&#xff0c;对事件因果关系的研究主要分为两类任务&#xff1a;事件因果关系识别&#xff08;Event Causality Identification&#xff0c;ECI&#xff09;和事件因果关系抽取&#xff08;Event Causality Extraction&#xff09;。事件因果关系识别旨在检测文本中两…

v5 实现动态时移播放

背景 有用户提出需要从当前时间前一段时间开始播放&#xff0c;比如 10s 前开始播放&#xff0c;或者 1 分钟前开始播放等。 在 v4 中有一个时光回溯功能&#xff0c;可以在配置中指定缓存时间&#xff0c;然后播放时可以指定 submode: 2来播放。 但是弊端是无法动态指定时间…

MySQL实训

项目名称与项目简介 股票交易系统是一个综合性的金融服务平台&#xff0c;它提供了股票买卖、交易查询、用户管理、股票信息管理以及资金账户管理等功能。系统旨在为用户提供一个安全、高效、便捷的股票交易环境&#xff0c;让用户能够实时掌握市场动态&#xff0c;做出合理的…

使用模板方法设计模式封装 socket 套接字并实现Tcp服务器和客户端 简单工厂模式设计

文章目录 使用模板方法设计模式封装套接字使用封装后的套接字实现Tcp服务器和客户端实现Tcp服务器实现Tcp客户端 工厂模式 使用模板方法设计模式封装套接字 可以使用模块方法设计模式来设计套接字 socket 的封装 模板方法&#xff08;Template Method&#xff09;设计模式是一…