【C++】stack queue

news2025/1/4 19:50:27

stack & queue

  • 一、容器适配器
  • 二、deque(了解)
  • 三、stack
    • 1. stack 的介绍
    • 2. 模拟实现 stack
  • 四、queue
    • 1. queue 的使用
    • 2. 模拟实现 queue
    • 3. priority_queue
      • (1)priority_queue 的介绍
      • (2)priority_queue 的使用
      • (3)仿函数
      • (4)模拟实现 priority_queue

一、容器适配器

适配器是一种设计模式(设计模式是一套被反复使用的、多数人知晓的、经过分类编目的、代码设计经验的总结),该种模式是将一个类的接口转换成我们希望的另外一个接口。

虽然 stackqueue 中也可以存放元素,但在 STL 中并没有将其划分在容器的行列,而是将其称为容器适配器,这是因为 stackqueue 只是对其他容器的接口进行了包装,STLstackqueue 默认使用 deque(后面介绍), 比如:

在这里插入图片描述

在这里插入图片描述

其实容器适配器就是复用其他容器,利用其他容器的功能来适配出一个新的容器。

二、deque(了解)

deque(双端队列): 是一种双开口的 “连续” 空间的数据结构,双开口的含义是:可以在头尾两端进行插入和删除操作,且时间复杂度为 O(1),与 vector 比较,头插效率高,不需要搬移元素;与 list 比较,空间利用率比较高。如果需要高效的随机存取,还要大量的首尾的插入删除则建议使用 deque.

deque 并不是真正连续的空间,而是由一段段连续的小空间拼接而成的,实际 deque 类似于一个动态的二维数组,其底层结构如下图所示:

在这里插入图片描述

vector 比较,deque 的优势是:头部插入和删除时,不需要搬移元素,效率特别高,而且在扩容时,也不需要搬移大量的元素,因此其效率是比 vector 高的;

list 比较,其底层是连续空间,空间利用率比较高,不需要存储额外字段;

但是,deque 有一个致命缺陷:不适合遍历,因为在遍历时,deque 的迭代器要频繁的去检测其是否移动到某段小空间的边界,导致效率低下,而序列式场景中,可能需要经常遍历,因此在实际中,需要线性结构时,大多数情况下优先考虑 vectorlistdeque 的应用并不多,而目前能看到的一个应用就是,STL 用其作为 stackqueue 的底层数据结构。

那么为什么选择 deque 作为 stackqueue 的底层默认容器呢?

stack 是一种后进先出的特殊线性数据结构,因此只要具有push_back()pop_back() 操作的线性结构,都可以作为 stack 的底层容器,比如 vectorlist 都可以;

queue先进先出的特殊线性数据结构,只要具有 push_backpop_front 操作的线性结构,都可以作为 queue 的底层容器,比如 list。但是 STL 中对 stackqueue 默认选择 deque 作为其底层容器,主要是因为:

  1. stackqueue 不需要遍历 (因此stack和queue没有迭代器),只需要在固定的一端或者两端进行操作。
  2. stack 中元素增长时,dequevector 的效率高(扩容时不需要搬移大量数据);queue 中的元素增长时,deque 不仅效率高,而且内存使用率高。结合了 deque 的优点,而完美的避开了其缺陷。

三、stack

1. stack 的介绍

我们先可以看一下 stack 的文档介绍:stack.

  1. stack 是一种容器适配器,专门用在具有后进先出操作的上下文环境中,其删除只能从容器的一端进行元素的插入与提取操作。

  2. stack 是作为容器适配器被实现的,容器适配器即是对特定类封装作为其底层的容器,并提供一组特定的成员函数来访问其元素,将特定类作为其底层的,元素特定容器的尾部(即栈顶)被压入和弹出。

  3. stack 的底层容器可以是任何标准的容器类模板或者一些其他特定的容器类,这些容器类应该支持以下操作:

     		empty:判空操作
     		back:获取尾部元素操作
     		push_back:尾部插入元素操作
     		pop_back:尾部删除元素操作
    
  4. 标准容器 vector、deque、list 均符合这些需求,默认情况下,如果没有为 stack 指定特定的底层容器,默认情况下使用 deque.

我们先简单地看看 stack 的使用:

		void test_stack()
		{
			stack<int> st;
			st.push(1);
			st.push(2);
			st.push(3);
			st.push(4);
			st.push(5);
			while (!st.empty())
			{
				cout << st.top() << ' ';
				st.pop();
			}
			cout << endl;
		}

运行结果如下:

在这里插入图片描述

2. 模拟实现 stack

我们使用 deque 作为 stack 的适配器模拟实现:

		#pragma once
		#include <vector>
		#include <deque>
		
		namespace Young
		{
			template <class T, class Container = deque<T>>
			class Stack
			{
			public:
				// 入栈
				void push(const T& val)
				{
					_con.push_back(val);
				}
		
				// 出栈
				void pop()
				{
					_con.pop_back();
				}
		
				// 获取栈顶元素
				T& top()
				{
					return _con.back();
				}
		
				// const 对象获取栈顶元素
				const T& top() const
				{
					return _con.back();
				}
				
				// 获取栈的大小
				size_t size()
				{
					return _con.size();
				}
		
				// 判断栈是否为空
				bool empty()
				{
					return _con.empty();
				}
		
			private:
				Container _con;
			};
		}

如上,stack 的常用接口就实现好了,我们再用我们自己实现的 stack 测试一下:

在这里插入图片描述

四、queue

1. queue 的使用

我们先看一下 queue 的文档介绍:queue.

  1. 队列是一种容器适配器,专门用于在 FIFO 上下文(先进先出)中操作,其中从容器一端插入元素,另一端提取元素。

  2. 队列作为容器适配器实现,容器适配器即将特定容器类封装作为其底层容器类,queue 提供一组特定的成员函数来访问其元素。元素从队尾入队列,从队头出队列。

  3. 底层容器可以是标准容器类模板之一,也可以是其他专门设计的容器类。该底层容器应至少支持以下操作:

     		empty:检测队列是否为空
     		size: 返回队列中有效元素的个数
     		front:返回队头元素的引用
     		back: 返回队尾元素的引用
     		push_back:在队列尾部入队列
     		pop_front:在队列头部出队列
    
  4. 标准容器类 dequelist 满足了这些要求。默认情况下,如果没有为 queue 实例化指定容器类,则使用标准容器 deque.

先简单看一下 queue 的使用:

		void test_queue()
		{
			queue<int> q;
			q.push(1);
			q.push(2);
			q.push(3);
			q.push(4);
			q.push(5);
			while (!q.empty())
			{
				cout << q.front() << ' ';
				q.pop();
			}
			cout << endl;
		}

运行结果如下:

在这里插入图片描述

2. 模拟实现 queue

我们也使用 deque 适配 queue

		#pragma once
		#include <deque>
		
		
		namespace Young
		{
			template<class T, class Container = deque<T>>
			class Queue
			{
			public:
				// 入队列
				void push(const T& val)
				{
					_con.push_back(val);
				}
		
				// 出队列
				void pop()
				{
					_con.pop_front();
				}
		
				// const 对象获取队头元素
				const T& front() const
				{
					return _con.front();
				}
		
				// 获取队头元素
				T& front()
				{
					return _con.front();
				}
		
				// const 对象获取队尾元素
				const T& back() const
				{
					return _con.back();
				}
		
				// 获取队尾元素
				T& back()
				{
					return _con.back();
				}
		
				// 获取队列长度
				size_t size()
				{
					return _con.size();
				}
		
				// 判断队列是否空
				bool empty()
				{
					return _con.empty();
				}
			private:
				Container _con;
			};
		}

我们再使用自己实现的 queue,来测试一下:

在这里插入图片描述

3. priority_queue

(1)priority_queue 的介绍

priority_queue:优先级队列,是属于队列的一种,我们先看一下它的文档介绍 priority_queue.

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

  2. 此上下文类似于堆,在堆中可以随时插入元素,并且只能检索最大堆元素(优先级队列中位于顶部的元素)。

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

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

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

(2)priority_queue 的使用

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

在这里插入图片描述

注意,我们看文档,默认情况下 priority_queue大堆,但是上图中红框中的是一个仿函数(后面介绍),就是实现比较的,其中 less 有小的意思,但是它却实现成大堆,这里要注意。其次,我们使用默认的参数时,只需要传第一个参数即可,后面的使用缺省参数即可,但是我们需要使用小堆的时候就需要将全部参数都传进去;我们先来看看使用:

		#include <vector>
		#include <queue>
		#include <functional> // greater算法的头文件
		void TestPriorityQueue()
		{
			// 默认情况下,创建的是大堆,其底层按照小于号比较
			vector<int> v{ 3,2,7,6,0,4,1,9,8,5 };
		
			priority_queue<int> q1;
			for (auto& e : v)
				q1.push(e);
		
			cout << q1.top() << endl;
		
		
			// 如果要创建小堆,将第三个模板参数换成 greater 比较方式
			priority_queue<int, vector<int>, greater<int>> q2(v.begin(), v.end());
			cout << q2.top() << endl;
		}

运行结果如下:

在这里插入图片描述

(3)仿函数

仿函数又称为函数对象,是一个能行使函数功能的类。它的使用和我们平时的函数调用一样。

首先我们得先实现一个类,这个类中需要实现 () 运算符重载,里面实现的功能需要我们自己实现,假设我们需要实现 priority_queue 中的大堆,如下:

		// 仿函数 --- 大堆,大的优先级大
		template <class T>
		class Less
		{
		public:
			bool operator()(const T& x, const T& y)
			{
				return x < y;
			}
		};

那么如何调用呢?首先我们得先创建一个对象,再使用这个对象进行调用函数:

			Less<int> less;
			cout << less(1, 8) << endl; // 与下等价
			cout << less.operator()(1, 8) << endl;

运行结果如下:

在这里插入图片描述

下面我们使用仿函数的形式模拟实现 priority_queue.

(4)模拟实现 priority_queue

	#pragma once
	#include <vector>
	
	namespace Young
	{
		// 模板参数
		template <class T, class Container = vector<T>, class Compare = Less<T>>
		class PriorityQueue
		{
		public:
			// 向上调整 --- 大堆
			void adjust_up(size_t child)
			{
				Compare com;
				size_t parent = (child - 1) / 2;
				while (child > 0)
				{
					//if (_con[parent] < _con[child]) // 与下等价
					if (com(_con[parent], _con[child]))
					{
						swap(_con[child], _con[parent]);
						child = parent;
						parent = (child - 1) / 2;
					}
					else
					{
						break;
					}
				}
			}
	
			// 向下调整 --- 大堆
			void adjust_down(size_t parent)
			{
				Compare com;
				size_t child = parent * 2 + 1;
				
				while (child < _con.size())
				{
					if (child + 1 < _con.size() && _con[child + 1] > _con[child])
					{
						++child;
					}
	
					// if (_con[parent] < _con[child])  // 与下等价
					if (com(_con[parent], _con[child]))
					{
						swap(_con[child], _con[parent]);
						parent = child;
						child = parent * 2 + 1;
					}
					else
					{
						break;
					}
				}
			}
	
			// 入数据
			void push(const T& val)
			{
				_con.push_back(val);
				adjust_up(_con.size() - 1);
			}
	
			// 出数据
			void pop()
			{
				swap(_con[0], _con[_con.size() - 1]);
				_con.pop_back();
	
				adjust_down(0);
			}
	
			// 获取优先级最大的数据
			const T& top()
			{
				return _con.front();
			}
	
			// 判空
			bool empty()
			{
				return _con.empty();
			}
	
		private:
			Container _con;
		};
	}

下面我们使用自己实现的优先级队列进行测试:

		void test_priority_queue()
		{
			// Young::PriorityQueue<int, vector<int>, Greater<int>> pq; // 小堆
		
			// Young::PriorityQueue<int, vector<int>, Less<int>> pq; // 大堆,与下等价
			Young::PriorityQueue<int> pq;  // 缺省参数默认是大堆
			pq.push(2);
			pq.push(8);
			pq.push(1);
			pq.push(0);
			pq.push(10);
		
			while (!pq.empty())
			{
				cout << pq.top() << ' ';
				pq.pop();
			}
		
			cout << endl;
		}

测试结果如下:

在这里插入图片描述

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

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

相关文章

数字IC笔试千题解--单选题篇(二)

前言 出笔试题汇总&#xff0c;是为了总结秋招可能遇到的问题&#xff0c;做题不是目的&#xff0c;在做题的过程中发现自己的漏洞&#xff0c;巩固基础才是目的。 所有题目结果和解释由笔者给出&#xff0c;答案主观性较强&#xff0c;若有错误欢迎评论区指出&#xff0c;资料…

如何通过Python实现接口自动化的参数关联?

前言 通常在接口自动化中&#xff0c;经常会参数关联的问题&#xff0c;那么什么是参数关联&#xff1f; 参数关联就是上一个接口的返回值会被下一个接口当做参数运用&#xff0c;其中Python中可以实现参数关联的方法有很多种&#xff0c;今天小编给大家介绍下&#xff0c;如…

IC验证| Verilog语法详解之条件语句

Verilog 是一种用于数字逻辑电路设计的硬件描述语言&#xff0c;可以用来进行数字电路的仿真验证、时序分析、逻辑综合。 既是一种行为级&#xff08;可用于电路的功能描述&#xff09;描述语言又是一种结构性&#xff08;可用于元器件及其之间的连接&#xff09;描述语言。 …

RedisTemplate出现\xac\xed\x00\x05t\x00\x0f前缀解决

问题描叙 出现这种乱码前缀的原因是没有进行序列化&#xff0c;因此导致在传输过程出现乱码问题&#xff0c;存到数据库&#xff0c;发现 key,hash key/value 都有 \xAC\xED\x00\x05t\x00 前缀。RedisTemplate类中默认是没有设置序列化的。 解决方法 设置RedisTemplate的序列…

如何快速建立一个专业、高效的宠物医院小程序?

随着社会的发展和科技的进步&#xff0c;人们对于宠物的关注度越来越高&#xff0c;养宠物已经成为了许多人的生活方式。然而&#xff0c;宠物的健康问题也随之而来&#xff0c;宠物医院成为了不可或缺的存在。为了更好地服务于宠物主人&#xff0c;打造一个专属的宠物医院线上…

IntelliJ IDEA 左侧Commit栏不见了

1.点击File->Settings->Version Control->Commit 2.勾选Use non-modal commit interface

关于ElementUI之首页导航与左侧菜单实现

目录 一.Mock 1.1.什么是Mock.js 1.2.特点 1.3.安装与配置 1.3.1. 安装mock.js 1.3.2.引入mock.js 1.4.mockjs使用 1.4.1.定义测试数据文件 1.4.2.mock拦截Ajax请求 1.4.3.界面代码优化 二.总线 2.1.是什么 2.2.前期准备 2.3.配置组件与路由关系 2.3.1. 配置组件 …

Mock.js之Element-ui搭建首页导航与左侧菜单

&#x1f3ac; 艳艳耶✌️&#xff1a;个人主页 &#x1f525; 个人专栏 &#xff1a;《Spring与Mybatis集成整合》《springMvc使用》 ⛺️ 生活的理想&#xff0c;为了不断更新自己 ! 1、Mock.js的使用 1.1.什么是Mock.js Mock.js是一个模拟数据的生成器&#xff0c;用来帮助前…

希尔排序代码及时间空间复杂度

希尔排序&#xff08;Shell Sort&#xff09;是一种插入排序的改进算法&#xff0c;它通过将数据分成多个小组来排序&#xff0c;然后逐渐减小这些小组的间隔&#xff0c;直到最后一次使用标准的插入排序算法。希尔排序的时间复杂度取决于使用的间隔序列&#xff0c;通常为 O(n…

Vulnhub-driftingbules:5 靶机复现完整过程

记录对driftingbules:5 靶机的复现过程 kali的IP地址&#xff1a;192.168.200.14 靶机IP地址&#xff1a;192.168.200.60 一、信息收集 1.对利用nmap目标靶机进行扫描 由于arp-scan属于轻量级扫描&#xff0c;在此直接使用nmap进行对目标靶机扫描开放端口 nmap -A -p 1-65…

从SmartPay dll学到的内容 宏定义 单件模式 迭代 日志记录函数进入与出来

日志记录函数进入与出来&#xff1a;利用C的反初始化来记录退出 函数运行记时、调用次数统计等 宏定义 配置里的宏 WIN32;NDEBUG;_WINDOWS;_USRDLL;SMARTPAY_PGLDLL_192787_EXPORTS;ESLOG_RELEASE;HAVE_STRUCT_TIMESPEC;%(PreprocessorDefinitions)减少代码耦合 关闭日志等 …

国内音视频开发的前景怎么样?

国内音视频开发的前景怎么样? 本人就是音视频开发&#xff0c;谈一下我的观点。 目前干我们这一行的年纪都比较大&#xff0c;我自己工作五年就是很年轻的了。年会上老板说除了音视频中心的大家都是比较年轻的。。。 有些也是过了35岁了&#xff0c;四十的都有。是不是觉得这…

Unity(四) 基于关键帧的动画与骨骼动画

Unity中有两种类型的动画&#xff1a;基于关键帧的动画和骨骼动画 基于关键帧的动画是最常见的动画形式&#xff0c;也称为帧动画。它将每一帧的动画存储为一个离散的关键帧&#xff0c;然后通过计算每一帧之间的差异来创建动画。这种类型的动画适用于不需要太多交互或程序控制…

【network】丢包网络情况记录

目录 2023-09-25 09:20 2023-09-25 11:23 接口流量 2023-09-25 09:20 2023-09-25 11:23 接口流量

使用自功率谱、互功率谱估计滤波器幅频特性

这段时间终于对工程中的随机信号的一般处理方式有点头绪了&#xff0c;功率谱密度估计是十分重要的方式之一&#xff0c;仍需继续深入细化相关内容。 示例&#xff1a;使用自功率谱、互功率谱估计滤波器幅频特性&#xff0c;自己实现 & Matlab自带函数实现。 clc;clear;cl…

BUUCTF [BJDCTF2020]EasySearch 1

“.swp” 后缀通常用于表示 Vim&#xff08;一种文本编辑器&#xff09;的交换文件。Vim 是一个强大的文本编辑器&#xff0c;它在编辑文件时会创建交换文件以确保文件内容的安全性。 审阅 访问 index.php.swp 得到源码 <?phpob_start();function get_hash(){$chars ABC…

【PDF】pdf 学习之路

PDF 文件格式解析 https://www.cnblogs.com/theyangfan/p/17074647.html 权威的文档&#xff1a; 推荐第一个连接&#xff1a; PDF Explained &#xff08;译作《PDF 解析》&#xff09; | PDF-Explained《PDF 解析》https://zxyle.github.io/PDF-Explained/ https://zxyle…

缓冲区溢出漏洞预防

什么是缓冲区溢出 组成所有应用程序的程序由缓冲区组成&#xff0c;缓冲区是在内存中分配的临时空间&#xff0c;用于保存数据&#xff0c;直到它们移动到程序的其他部分&#xff0c;缓冲区可以包含的数据字节数最初将在代码开发期间指定&#xff0c;由于没有任何类型的边界检…

UNet网络训练

UNet网络训练 训练资源 构建好UNet网络模型后&#xff0c;需要进行训练。但是训练需要特别多的原始图像和标签图像&#xff0c;对于一般而言这一步特别繁琐&#xff0c;不过在网上有一些免费的数据集可以让我们省略这一步&#xff0c;直接进行训练测试。 VOC&#xff08;Visu…

Centos环境使用Docker安装Kafka

1 Kafka简介 1、kafka是什么&#xff1f; Kafka是一种高吞吐量的分布式发布订阅消息系统&#xff0c;它可以处理消费者规模的网站中的所有动作流数据&#xff0c;具有高性能、持久化、多副本备份、横向扩展能力。 2、kafka的工作原理[去耦合] Kafka采用的是订阅-发布的模式&am…