C++ stack与queue的使用与简单实现

news2025/1/8 4:53:10

 

目录

0. 适配器

1. stack的简要介绍

 2. stack的简单使用

3. queue的简要介绍

 4. queue的简单使用

 STL标准库中stack和queue的底层结构

 deque简单介绍

5. stack的模拟实现

6. queue的模拟实现


0. 适配器

在文章开始前我们先了解一下适配器的概念

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

1. stack的简要介绍

是一个容器适配器,提供了后进先出(或先进后出)的数据结构,其元素仅能从容器的一端插入和提取。

使用特定容器类的封装对象作为其底层容器的类,提供一组特定的成员函数来访问其元素。元素从特定容器的背面出栈

 2. stack的简单使用

函数说明接口说明
stack()构造空的栈
empty()检测stack
size()返回stack中元素的个数
top()返回栈顶元素的引用
push()将元素val压入stack中
pop()将stack中最后入栈的元素弹出

如下代码所示

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

int main()
{
	stack<int> st;
	st.push(1);//往栈里压入元素
	st.push(12);
	st.push(123);
	st.push(1234);

	while (!st.empty())//如果栈里没有元素了就停止循环
	{
		cout << "栈里的元素个数为: " << st.size() << "  栈顶的元素为:  " << st.top() << endl;
		st.pop();//删除栈里的元素,后进来的先出去,所以先删1234
	}
    return 0;
}

 输出结果如下图所示

 验证了我们所说的后入栈的先出栈,而栈顶的元素就是最后入栈的元素(如果入栈过程中没有元素提前pop出栈)。

3. queue的简要介绍

队列是一种容器适配器,专门用于先进先出中操作,从容器一端插入元素,另一端提取元素。

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

 4. queue的简单使用

函数声明接口说明
queue()构造空的队列
empty()检测队列是否为空,是返回true,否则返回false
size()返回队列中有效元素的个数
front()返回队头元素的引用
back()返回队尾元素的引用
push()在队尾将元素val入队
pop()将队头元素出队列

标准容器类deque和list满足了这些要求,如果我们没为queue实例化指定容器类(没使用下方定义),则其默认使用deque。

queue<int,list<int>> qu;

代码如下

#include<iostream>
#include<stack>
#include<queue>
#include<list>
using namespace std;

int main()
{
    //queue<int,list<int>> qu;
	queue<int> qu;
	qu.push(1);//元素入队
	qu.push(12);
	qu.push(123);
	qu.push(1234);

	while (!qu.empty())//如果队列里里没有元素了就停止循环
	{
		cout << "队列里的元素个数为: " << qu.size() << "  队头的元素为:  " << qu.front()<<"  队尾的元素为:  " << qu.back()<< endl;
		qu.pop();//删除队列里的元素,先进来的先出去,所以先删1
	}
	return 0;
}

输出结果如下

可以直观的看到,先入队的队头元素先出队了,后入队的队尾元素后出队 


 STL标准库中stack和queue的底层结构

虽然stack和queue中也可以存放元素,但在STL中没有将其划分在容器的行列,而是将其称为容器适配器,这是因为stack和queue只是对其他容器的接口进行了包装,STL中stack和queue默认使用deque

 deque简单介绍

deque(双端队列):是一种双开口的“连续”空间的数据结构,双开口的含义是: 可以在头尾两端进行插入和删除操作,且时间复杂度为O(1),与vector比较,头插效率高不需要搬移元素与list比较空间利用率比较高

 其并不是真正的连续的空间,而是由一段段连续的小空间拼接而成,实际deque类似一个动态的二维数组

双端队列底层是一段假想的连续空间,实际是分段连续的,为了维护其“整体连续”以及随机访问的假象,就交给deque的迭代器了,所以deque的迭代器设计就比较复杂

迭代器元素与功能大致有

1. cur :用来遍历当前数组

2. first :指向一个数组的头

3. last :指向一个数组的尾

4. node :指向一个数组

 优:

  1. 与vector相比:头插(如果前面没有空间了,就在前面再开一个数组,将元素插入新开辟数组的最后的位置)和尾删时,不需要搬移元素,效率特别高,而且在扩容时也不需要搬移大量的元素,因此效率比vector高
  2. 与list相比:其底层有连续空间,空间利用率比较高,不需要存储额外字段

劣:

不适合遍历,在遍历时deque的迭代器要去频繁地检测其是否移动到某段小空间的边界,导致效率低下,而有些场景中可能经常需要遍历,因此在实际中,需要线性结构时,大多数情况下优先考虑vector和list,deque的应用并不多,STL用其作为stack和queue的底层数据结构。

5. stack的模拟实现

默认使用deque的原因

1. 头插头删效率极高,又不需要遍历

 2. 扩容不用搬移元素,比vector强

实现非常简单,我们用容器自带的接口来实现即可

#include<iostream>
#include<deque>
#include<list>
#include<vector>
using namespace std;
namespace Pc
{

	//template<class T, class container = vector<T> >
	//template<class T, class container = list<T> >
	template<class T, class container = deque<T> >
	class stack
	{
	public:

		stack()
		{

		}
		void push(const T& x)
		{
			_sta.push_back(x);
		}
		void pop()
		{
			_sta.pop_back();
		}
		T& top()
		{
			return _sta.back();//传最后的引用返回
		}
		const T& top() const
		{
			return _sta.back();//传最后的引用返回
		}
		size_t size() const
		{
			return _sta.size();
		}
		bool empty()
		{
			return _sta.empty();
		}
	private:
		container _sta;
	};
}

由于我们上面的push_back、pop_back、size等无论list、vector还是deque都有所以这三个容器都可以实现

6. queue的模拟实现

默认使用deque的原因

1. 依然是不需要遍历,只需要尾插头删(vector没有头删)

2. deque元素在内存中相对集中,减少了缓存未命中的次数,提高程序运行效率。list元素太过分散,缓存命中率低

 实现代码如下

#include<iostream>
#include<deque>
#include<list>
#include<vector>
using namespace std;
namespace Pc
{

	//template<class T, class container = list<T> >
	template<class T, class container = deque<T> >
	class queue
	{
	public:

		queue()
		{

		}
		void push(const T& x)
		{
			_que.push_back(x);
		}
		void pop()
		{
			_que.pop_front();
		}
		T& back()
		{
			return _que.back();//传最后的引用返回
		}
		const T& back() const
		{
			return _que.back();//传最后的引用返回
		}
		T& front()
		{
			return _que.front();//传最后的引用返回
		}
		const T& front() const
		{
			return _que.front();//传最后的引用返回
		}
		size_t size() const
		{
			return _que.size();
		}
		bool empty()
		{
			return _que.empty();
		}
	private:
		container _que;
	};
}

这篇就到这里啦~(づ ̄3 ̄)づ╭❤~

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

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

相关文章

【第二节】80x86汇编-寄存器和标志位

目录 前言 一、汇编相关概念 1.1 数据表示与类型 1.2 汇编语言的构成 1.3 存储器及指令、数据 1.4 存储单元 1.5 CPU对存储器的读写操作 1.6 CPU读写内存单元的过程 1.7 intel CPU发展 1.8 8086 内部结构 二、寄存器 2.1 寄存器概览 2.2 32位寄存器 2.3 16位寄存器…

浅谈C语言预处理

文章目录 预处理1、预定义符号2、#define定义标识符和宏A、#define定义标识符B、#define定义宏a、宏的定义b、宏的使用c、宏和函数 3、条件编译4、头文件包含A、两种包含形式B、防止头文件被重复包含 预处理 什么是预处理&#xff1f;预处理是C语言编译的三个过程&#xff08;…

一口气把halcon的所有运算符说清楚

halcon的运算符大体分以下几类 一&#xff1a;赋值运算符&#xff1a; (1) 赋值(:)&#xff08;左边的赋值给右边&#xff09; 二&#xff1a;算术运算符 (1)加()、减(-)、乘(*)、除(/)、求余(%) 三&#xff1a;关系运算符 (3)#、&#xff01; 不等于 四&#xff1a;逻辑运…

java中运算符的详细知识点

算数运算符 a 先赋值再加1 a 先加1在赋值 --的道理是一样的 赋值运算符 1. - * / % 当两侧数据类型不一致时&#xff0c;可以使用自动类型转换或使用 强制类型转换原则 支持连续赋值 - * / % 不会改变基础类型 测试一下&#xff1a; 比较运算符 运算结果为布尔类型 &#x…

Windows上安装WSL,学习Linux

1. 什么是WSL 先说大白话WSL就是让Windows不安装虚拟机可以额外拥有Linux操作系统&#xff0c;以供学习和测试 WSL&#xff08;Windows Subsystem for Linux&#xff09;和WSL2 是微软推出的两个工具&#xff0c;旨在让用户能够在 Windows 操作系统上运行 Linux 的命令行工具…

MySQL数据分析进阶(十四)保护数据库

※食用指南&#xff1a;文章内容为‘CodeWithMosh’SQL进阶教程系列学习笔记&#xff0c;笔记整理比较粗糙&#xff0c;主要目的自存为主&#xff0c;记录完整的学习过程。&#xff08;图片超级多&#xff0c;慎看&#xff01;&#xff09; 【中字】SQL进阶教程 | 史上最易懂S…

Unity游戏开发004:如何在Unity中对物体进行基本操作

Unity游戏开发 “好读书&#xff0c;不求甚解&#xff1b;每有会意&#xff0c;便欣然忘食。” 本文目录&#xff1a; Unity游戏开发 Unity游戏开发前言左侧工具栏概述1. **创建物体**2. **移动&#xff08;Move&#xff09;**3. **旋转&#xff08;Rotate&#xff09;**4. **缩…

注册数据查询工具

注册数据查询工具&#xff1a;ICANN Lookup

微前端插件 v-micro-app-plugin

v-micro-app-plugin 是一款基于京东MicroApp框架的微前端插件&#xff0c;旨在帮助开发者快速地将微应用集成到不同的系统中&#xff0c;实现高效、灵活的前端模块化开发。以下是详细的使用指南&#xff0c;帮助你快速上手。 微前端插件 v-micro-app-plugin 源码地址&#xff1…

golang 命令行聊天室

实战简介&#xff1a; 基于tcp协议实现功能 服务器端&#xff1a; 接受用户消息和循环转发 对功能命令进行处理&#xff08;meun查询功能词&#xff0c;changeName改名&#xff0c;online查询在线人数&#xff0c;quit退出&#xff09; 客户端&#xff1a; 接受服务器发送…

【话题】程序员之路:高效编码与持续成长的平衡之道

目录 程序员如何平衡日常编码工作与提升式学习&#xff1f; 引言 方向一&#xff1a;高效编码习惯与时间管理技巧 方向二&#xff1a;提升式学习的策略 实际案例&#xff1a; 方向三&#xff1a;职业发展与个人成长的和谐共生 实际案例&#xff1a; 程序员如何平衡日常编码工作…

金融业与三方公司网络互联方法

文章目录 云桌面定义类型非持久桌面主要特点和优点 持久桌面主要特点和优点 网络缺点 专线定义特点网络缺点 VPN定义特点网络缺点 零信任定义优点缺点与挑战网络应用最佳实践案例与趋势 互联网加白名单定义应用场景访问流程图优点缺点相关技术 云桌面 定义 云桌面是一种基于云…

工厂间的库存转储

从一个工厂到另一个工厂实施实物物料转储&#xff1a; 一步式程序两步式程序&#xff1a;使用两步式程序进行库存转储可能是因为两个工厂相距甚远&#xff0c;货物在某个特定时间内处于运输状态&#xff1b;或者两个工厂分别由不同的员工负责&#xff0c;每个员工只能过帐本工…

红酒与节日装饰:打造节日氛围的需备品

随着节日的脚步渐渐临近&#xff0c;节日的氛围也愈发浓厚。在这个特殊的时刻&#xff0c;红酒与节日装饰无疑成为了营造节日氛围的需备品。洒派红酒&#xff08;Bold & Generous&#xff09;作为定制红酒的品牌&#xff0c;其不同的韵味与节日装饰的精致整合&#xff0c;共…

react笔记:redux

redux状态管理 安装redux&#xff1a;num i redux 新建redux文件夹&#xff1a; store.jscount_reducer.js count_action.js constant.js (常量&#xff09; 1. store.js文件&#xff1a; // 该文件专门用于暴露一个store对象&#xff0c;整个应用只有一个store对…

MyBatis巧用1=0,再鸡肋也得用

一、问题导向 今天在实现文件上传数据库测试中&#xff0c;发现了一个略微容易被问题。虽然业务层可以约束空字符串的传入。但是测试DAO数据层时&#xff0c;被同事发现&#xff0c;说我&#xff1a;“为什么不加10”。 请看问题。 如果用户账号为空&#xff0c;则有分页输出…

在JavaFx中使用StyledLayerDescriptor(SLD)1.0和1.1版本进行地图美化实践

目录 前言 一、最开始的初始世界 1、默认的样式 二、注册事件让地图实现交互 1、绑定事件及注册处理逻辑 三、地图美化&#xff0c;让地图生动起来 1、POM.xml中引入相应的依赖 2、GeoTools加载不同版本的SLD问题 3、加载1.1.0版本的SLD 四、总结 前言 俗话说&#xff…

超简单理解LSTM和GRU模型

目录 参考资料 RNN在反向传播时容易遭受梯度消失的问题&#xff0c;而梯度是用于更新神经网络权重的关键因子&#xff0c;梯度消失描述的是梯度在时间序列反向传播中逐渐减小的情况。 若梯度过小&#xff0c;它对于网络的学习贡献甚微。结果是&#xff0c;在RNN中&#xff0c;梯…

成功的秘诀:客户推荐推动 SaaS 成功的4个原因

在竞争激烈的SaaS市场中&#xff0c;脱颖而出并非易事。客户推荐合作伙伴关系正是这样一种强大的营销渠道&#xff0c;它利用口碑传播的力量&#xff0c;将满意的客户转变为品牌倡导者&#xff0c;从而显著提升品牌信誉和市场影响力。 您的 SaaS 品牌应该接受客户推荐营销的4 …

进 程

1.进程&#xff1a;进行中的程序。 微观串行&#xff0c;宏观并行。 程序的一次执行过程 进程是程序的一个实例 一个程序可以对应一个或多个进程。 2.为什么需要进程&#xff1f; 3.进程的组成部分&#xff1a; 进程 pcb 块 栈|堆|bss|data|text 其中&#xff1a; 家族…