priority_queue的使用以及模拟实现

前言

上一期我们对stack和queue进行了使用的介绍,以及对底层的模拟实现!以及容器适配器做了介绍,本期我们在来介绍一个容器适配器priority_queue!

本期内容介绍

priority_queue的使用

仿函数介绍

priority_queue的模拟实现

什么是priority_queue?

priority_queue称为优先级队列,实际上就是堆(如果忘了什么是堆, 请点击这里)!它也是容器适配器,底层默认的容器是vector,默认是大堆!

常用的接口介绍

empty

判断是否为空

size

元素的个数

top

获取堆顶元素

注意:堆顶元素是不允许修改的,如果修改了会影响整个堆的结构,所以用const修饰!!

push

插入一个新元素

pop

删除堆顶的元素

OK,用一下!

void test()
{
	priority_queue<int> pq;
	pq.push(2);
	pq.push(15);
	pq.push(-1);
	pq.push(90);

	size_t sz = pq.size();
	cout << sz << endl;

	bool empty = pq.empty();
	cout << empty << endl;

	int top = pq.top();
	cout << top << endl;

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

priority_queue在是很有用的,例如TopK问题和堆排序。堆排序不在介绍,在数据结构已经详细的介绍了,这里拿个题目来再次理解一下TopK:

数组中第K大元素

它的题目意思就是让你,找出第K大的元素,但是要求时间度杂度O(N)!

思路:这里有三种, 第三种最优。

1、利用排序数组,然后返回倒数k个元素即可!

2、借助堆

3、快速选择算法(算法专栏介绍)

我们这里就先介绍前两种!

1、利用排序数组,然后返回倒数k个元素即可!(时间复杂度不符合)

class Solution {
public:
    int findKthLargest(vector<int>& nums, int k) {
        sort(nums.begin(), nums.end());
        return nums[nums.size() -k ];
    }
};

这里虽然过了,但是这个代码的时间复杂度是O(N*logN),不符合!

2、借助堆

class Solution {
public:
    int findKthLargest(vector<int>& nums, int k) 
    {
        priority_queue<int> pq(nums.begin(), nums.end());//将数组的元素放到堆里面
        for(int i = k; i > 1; i--)//将前k-1个弹出堆
        {
            pq.pop();
        }

        return pq.top();//返回堆顶的元素
    }
};

OK,这就过了,但是时间复杂度也是不行的!也还是O(N*logN)。最优解在后续的算法专栏会介绍!这主要是主要是演示一下堆在OJ中的用法~!

仿函数介绍

仿函数是一种类,它的对象可以向函数一样被调用。他是一种可调用对象,可以向像函数一样接收参数并返回结果!通常情况,仿函数是通过一个类实现operator ()来实现的!

 例如,上面刚介绍的priority_queue:

这里的less就是一个仿函数!还有就是sort:

OK,举个仿函数的例子:

struct cmp
{
	bool operator()(const int& a, const int& b)
	{
		return a < b;
	}
};

//class cmp
//{
//public:
//	bool operator()(const int& a, const int& b)
//	{
//		return a < b;
//	}
//};

void test()
{
	cmp cm;
	int result1 = cm(3, 5);

	int result2 = cm.operator()(3, 5);

	int result3 = cmp()(3, 5);
}

第一种和第二种本质是同一种,第二种是第一种的显示调用,第三种是匿名对象调用!当然这里的struct可以是class但注意的是如果是class你必须吧权限设置为public!

仿函数的用途

仿函数的用途我目前碰到的有两种(后面遇见了在补充):

1、STL算法库中的某些算法来用 仿函数定义他们的行为!例如:sort等

2、容器的自定义行为!例如priority_queue指定大小堆!等

#include <algorithm>
template<class T>
struct Compare
{
	bool operator() (const T& a, const T& b)
	{
		return a > b;
	}
};

void test()
{	
	vector<int> v = { 1,3,0,12,43,5,9 };
	sort(v.begin(), v.end());//默认是升序
	for (const auto& e : v)
	{
		cout << e << " ";
	}
	cout << endl;

	sort(v.begin(), v.end(), Compare<int>());//降序  --》用匿名对象
	Compare<int> cmp;
	sort(v.begin(), v.end(), cmp);//降序 --》用实名对象
	for (const auto& e : v)
	{
		cout << e << " ";
	}
	cout << endl;
}

大堆和小堆

void test()
{
	vector<int> v = { 0,12,43,5,9 };
	priority_queue<int> pq1(v.begin(), v.end());//默认大堆
	while (!pq1.empty())
	{
		cout << pq1.top() << " ";
		pq1.pop();
	}
	cout << endl;

	priority_queue<int, vector<int>, greater<int>> pq2(v.begin(), v.end());//指定为小堆
	while (!pq2.empty())
	{
		cout << pq2.top() << " ";
		pq2.pop();
	}
	cout << endl;
}

这里priority_queue默认是less:

template <class T, class Container = vector<T>, class Compare = less<typename Container::value_type> > class priority_queue;

如果要切换为小堆,就得指定仿函数为greater,但是我们知道参数是倒着(从右往左)传递的,所以这里要指定为小堆的话,还要指定它的底层适配的容器~!一般是vector也可以是deque!!

priority_queue的模拟实现

priority_queue() 
{}

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

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

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

这几个不在多说,很简单直接调用默认容器的相关接口即可~!主要介绍一下这里的插入、删除和用用一段迭代器区间构造!

push

先插入到适配容器的尾部,然后进行向上调整!(向上调整不了解的点击上面介绍对那个文章的链接回忆一下)

另外注意的是这里我们不再是以前的大于小于比较了,而是用仿函数!

void adjust_up(size_t child)
{
	size_t parent = (child - 1) / 2;
	while (child > 0)
	{
		if (cmp()(_con[parent], _con[child]))//孩子比父亲大(小),交换
		{
			swap(_con[parent], _con[child]);
			child = parent;
			parent = (child - 1) / 2;
		}
		else
		{
			break;//否则,结束掉
		}
	}
}
void push(const T& val)
{
	_con.push_back(val);
	adjust_up(_con.size() - 1);
}

pop

先交换堆顶数据和最后一个数据!然后删除掉,堆顶数据,进行向下调整~!(向下调整和向上调整一样,详细见数据结构那里)

另外注意的是这里我们不再是以前的大于小于比较了,而是用仿函数!

void adjust_down(size_t parent)
{
	size_t child = parent * 2 + 1;//假设第一个孩子就是要交换的孩子
	while (child < _con.size())
	{
		if (child + 1 < _con.size() && cmp()(_con[child], _con[child + 1]))//假设错了
		{
			++child;//调整
		}

		if (cmp()(_con[parent], _con[child]))//孩子比父亲大(小)
		{
			swap(_con[parent], _con[child]);//交换
			parent = child;
			child = parent * 2 + 1;
		}
		else
		{
			break;//否则,结束掉
		}
	}
}
void pop()
{
	swap(_con.front(), _con.back());
	_con.pop_back();
	adjust_down(0);
}

迭代器区间构造

这里其实注意的就一点,当把数据给到适配器的容器后,要保证是堆结构!所以就要建堆,建堆的方式有两种(详见数据结构那里)向上调整建堆 和 向下调整建堆!

template<class Iterator>
priority_queue(Iterator first, Iterator last)
	:_con(first, last)
{
	int sz = _con.size();
	//向下调整建堆 O(N)
	for (int i = (sz - 1) / 2; i >= 0; i--)
	{
		adjust_down(i);
	}

	//向下调整建堆 O(N*logN)
	//for (int i = 1; i < sz; i++)
	//{
	//	adjust_up(i);
	//}
}

全部源码

#pragma once

#include <vector>

namespace cp
{
	template<class T>
	struct less
	{
		bool operator()(const T& a, const T& b)
		{
			return a < b;
		}
	};

	template<class T>
	struct greater
	{
		bool operator()(const T& a, const T& b)
		{
			return a > b;
		}
	};


	template<class T, class Con = std::vector<T>, class cmp =  less<T>>
	class priority_queue
	{
	public:
		priority_queue() 
		{}

		template<class Iterator>
		priority_queue(Iterator first, Iterator last)
			:_con(first, last)
		{
			int sz = _con.size();
			//向下调整建堆 O(N)
			for (int i = (sz - 1) / 2; i >= 0; i--)
			{
				adjust_down(i);
			}

			//向下调整建堆 O(N*logN)
			//for (int i = 1; i < sz; i++)
			//{
			//	adjust_up(i);
			//}
		}

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

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

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

		void push(const T& val)
		{
			_con.push_back(val);
			adjust_up(_con.size() - 1);
		}

		void pop()
		{
			swap(_con.front(), _con.back());
			_con.pop_back();
			adjust_down(0);
		}

	private:
		void adjust_up(size_t child)
		{
			size_t parent = (child - 1) / 2;
			while (child > 0)
			{
				if (cmp()(_con[parent], _con[child]))//孩子比父亲大(小),交换
				{
					swap(_con[parent], _con[child]);
					child = parent;
					parent = (child - 1) / 2;
				}
				else
				{
					break;//否则,结束掉
				}
			}
		}

		void adjust_down(size_t parent)
		{
			size_t child = parent * 2 + 1;//假设第一个孩子就是要交换的孩子
			while (child < _con.size())
			{
				if (child + 1 < _con.size() && cmp()(_con[child], _con[child + 1]))//假设错了
				{
					++child;//调整
				}

				if (cmp()(_con[parent], _con[child]))//孩子比父亲大(小)
				{
					swap(_con[parent], _con[child]);//交换
					parent = child;
					child = parent * 2 + 1;
				}
				else
				{
					break;//否则,结束掉
				}
			}
		}

	private:
		Con _con;
	};
}

OK,验证一下:

OK,没有问题!本期内容就分享到这里,好兄弟我们下期再见!

结束语:无人问津的日子,更应该加倍努力!

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

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

相关文章

2024年人工智能路线图

今天分享的是人工智能专题系列深度研究报告&#xff1a;《人工智能专题&#xff1a;2024年人工智能路线图》。 秘书制定部门的人工智能战略优先事项和政策&#xff0c;并且是关键的对话者与私营部门、联邦机构、州官员&#xff0c;以及主要的国际同行。这部长在白宫人力资源委员…

C/C++基础----运算符

算数运算符 运算符 描述 例子 两个数字相加 两个变量a b得到两个变量之和 - 两个数字相减 - * 两个数字相乘 - / 两个数字相除 - % 两个数字相除后取余数 8 % 3 2 -- 一个数字递减 变量a&#xff1a;a-- 、--a 一个数字递增 变量a: a 、 a 其中递…

如何在 7 天内掌握C++?

大家好&#xff0c;我是小康&#xff0c;今天我们来聊下如何快速学习 C 语言。 本篇文章适合于有 C 语言编程基础的小伙伴们&#xff0c;如果还没有学习过 C&#xff0c;请看这篇文章先入个门&#xff1a;C语言快速入门 引言&#xff1a; C&#xff0c;作为一门集面向过程和…

精彩回顾 | 「AI 驱动增长,研发数智化升级」分享沙龙成功举办

AI 应用元年&#xff0c;人工智能技术将如何助力企业发展新质生产力&#xff0c;构建增长动能&#xff1f; 日前&#xff0c;LigaAI 与深圳市企业联合会、西云数据联合举办了「AI 驱动增长&#xff0c;研发数智化升级」技术专题沙龙。本次活动围绕「AI」应用实践&#xff0c;邀…

2024-基于人工智能的药物设计方法研究-AIDD

AIDD docx 基于人工智能的药物设计方法研究 AI作为一种强大的数据挖掘和分析技术已经涉及新药研发的各个阶段&#xff0c;有望推动创新药物先导分子的筛选、设计和发现&#xff0c;但基于AI的数据驱动式创新药物设计和筛选方法仍存在若干亟待解决的问题。我们课题组的核心研究…

Kali中间人攻击

中间人攻击 中间人攻击&#xff08;Man-in-the-Middle Attack&#xff0c;简称MITM&#xff09;是一种网络安全攻击&#xff0c;其中攻击者插入自己&#xff08;作为“中间人”&#xff09;在通信的两个端点之间&#xff0c;以窃取或篡改通过的数据。攻击者可以监视通信&#x…

我为什么选择成为程序员?

前言&#xff1a; 我选择成为程序员不是兴趣所在&#xff0c;也不是为了职业发展&#xff0c;全是生活所迫&#xff01; 第一章&#xff1a;那年&#xff0c;我双手插兜&#xff0c;对外面的世界一无所知 时间回到2009年&#xff0c;时间过得真快啊&#xff0c;一下就是15年前…

多线程回答的滚瓜烂熟,面试官问我虚线程了解吗?我说不太了解!

Java虚拟线程&#xff08;Virtual Threads&#xff09;标志着Java在并发编程领域的一次重大飞跃&#xff0c;特别是从Java 21版本开始。这项新技术的引入旨在克服传统多线程和线程池存在的挑战。 多线程和线程池 在Java中&#xff0c;传统的多线程编程依赖于Thread类或实现Ru…

Green Hills 自带的MULTI调试器查看R7芯片寄存器

Green Hills在查看芯片寄存器时需要导入 .grd文件。下面以R7为例&#xff0c;演示一下过程。 首先打开MULTI调试器&#xff0c;如下所示View->Registers&#xff1a; 进入如下界面&#xff0c;选择导入寄存器定义文件.grd&#xff1a; 以当前R7芯片举例&#xff08;dr7f7013…

室内定位中文综述阅读

1 室内高精度定位技术总结与展望 [4]柳景斌,赵智博,胡宁松等.室内高精度定位技术总结与展望[J].武汉大学学报(信息科学 版),2022,47(07):997-1008.DOI:10.13203/j.whugis20220029. 1.1.1 WiFi‐RTT定位 2016 年 12 月&#xff0c;随着新版 IEEE802.11 标准的公布&#xff0c…

力扣55. 跳跃游戏

Problem: 55. 跳跃游戏 文章目录 题目描述思路复杂度Code 题目描述 思路 将题目稍作转化&#xff1a;验证最远走到的距离是否超出组数&#xff1b; 1.获取数组nums的长度n&#xff0c;定义int变量farthest初始化为0&#xff1b; 2.从0~n-1循环每次更新farthes的长度farthest …

2024年3月文章一览

2024年3月编程人总共更新了12篇文章&#xff1a; 1.2024年2月文章一览 2.Programming Abstractions in C阅读笔记&#xff1a;p308-p311 3.Programming Abstractions in C阅读笔记&#xff1a;p312-p326 4.Programming Abstractions in C阅读笔记&#xff1a;p327-p330 5.…

查询卖家已卖出的交易数据

要获取淘宝订单详情数据&#xff0c;你需要使用淘宝开放平台的API来获取数据。以下是获取淘宝订单详情数据的步骤&#xff1a; 在淘宝开放平台上创建一个应用&#xff0c;获取到AppKey和AppSecret。 使用OAuth 2.0授权方式&#xff0c;获取到授权码。 第三方公司授权 使用授…

快速排序(单边循环和双边循环)

快速排序 单边循环快排 pv指向分区中最后一个元素&#xff0c;i&#xff0c;j指向分区中第一个元素&#xff0c;j所指向的元素和pv指向的元素比较大小&#xff0c;如果比pv所指大&#xff0c;则j&#xff0c;否则与i所指元素交换位置&#xff0c;i&#xff0c;j&#xff1b;当…

雪亮工程视频联网综合管理/视频智能分析系统建设方案(二)

一、我国雪亮工程当前建设需求 1&#xff09;加强社会治安防控感知网络建设 加强社会治安防控智能感知网络建设&#xff0c;针对城中村、背街小巷、城乡结合部等重点区域建设安装视频监控设备&#xff0c;减少死角和盲区&#xff0c;与已有感知系统结合&#xff0c;形成高低搭…

树形查找试题(二叉树、红黑树)

一、单项选择题 01.对于二叉排序树&#xff0c;下面的说法中&#xff0c;()是正确的。 A.二叉排序树是动态树表&#xff0c;查找失败时插入新结点&#xff0c;会引起树的重新分裂和组合 B.对二叉排序树进行层序遍历可得到有序序列 C.用逐点插入法构造二叉排序树&#xff0c;若先…

Harmony鸿蒙南向驱动开发-MIPI DSI接口使用

功能简介 DSI&#xff08;Display Serial Interface&#xff09;是由移动行业处理器接口联盟&#xff08;Mobile Industry Processor Interface (MIPI) Alliance&#xff09;制定的规范&#xff0c;旨在降低移动设备中显示控制器的成本。它以串行的方式发送像素数据或指令给外…

Centos安装MySQL提示公钥尚未安装

一、问题 在Centos7.9使用yum安装MySQL时出现错误&#xff0c;提示&#xff1a;mysql-community-server-5.7.44-1.el7.x86_64.rpm 的公钥尚未安装&#xff0c;如下图所示&#xff1a; 执行命令&#xff1a;systemctl start mysqld也提示错误&#xff1a;Failed to start mysq…

目标:3100P 北京最大规模公共智算中心升级 开启建设加速度

近日&#xff0c;企商在线石景山智能算力中心升级备案获批&#xff0c;算力规模由原先的610P跃升至3100P&#xff0c;成为当前北京市规模最大的公共智算中心之一。 石景山智能算力中心效果图 在计算机领域&#xff0c;“P”代表“Petaflop”&#xff0c;1P即每秒1000万亿次浮点…

【JAVA基础篇教学】第三篇:Java循环控制语句

博主打算从0-1讲解下java基础教学&#xff0c;今天教学第三篇&#xff1a;Java循环控制语句。 在Java中&#xff0c;循环控制语句用于重复执行一段代码&#xff0c;直到满足特定条件为止。Java提供了多种类型的循环语句&#xff0c;包括for循环、while循环和do-while循环。 一…