深入STL之 栈与队列:数据结构探索之旅

📝个人主页🌹:Eternity._
⏩收录专栏⏪:C++ “ 登神长阶 ”
🤡往期回顾🤡:模拟实现list与迭代器
🌹🌹期待您的关注 🌹🌹

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

❀stack和queue

  • 📕1. stack和queue的基本概念
    • 🎩stack的基本概念
    • 🎈queue的基本概念
  • 📙2. stack与queue的常用操作
    • ⛰️stack的常用操作
    • 🌄queue的常用操作
  • 📚3. 容器适配器
  • 📒4. deque的简单介绍
    • 💧deque的原理介绍
    • 🌊deque的缺陷
  • 📜5. stack和queue的模拟实现
    • 🍂stack的模拟实现
    • 🍁queue的模拟实现
  • 📖6. priority_queue
    • 🌞priority_queue的基本概念
    • 🌙priority_queue的常用操作
    • ⭐priority_queue的模拟实现
  • 🔥7. 总结


前言: 在编程的世界里,数据结构是构建高效、可靠程序的基础。它们就像是我们编程工具箱中的精密工具,帮助我们解决各种复杂的问题。而在C++的STL中,栈(Stack)和队列(Queue)是两种非常重要的数据结构,它们以不同的方式管理和操作数据,为我们的程序提供了极大的灵活性

为了真正掌握它们,我们需要深入学习它们在STL中的实现方式,理解它们背后的原理和机制,以及学习如何在实际编程中有效地使用它们,让我们一起踏上学习STL栈与队列的旅程吧!


📕1. stack和queue的基本概念

🎩stack的基本概念

栈(Stack)是一种后进先出(LIFO)的数据结构

  • 它的操作特性使其在处理递归调用、函数调用栈以及撤销操作等问题时表现出色。通过栈,我们可以轻松地实现如括号匹配、表达式求值等算法。

在这里插入图片描述
在这里插入图片描述


🎈queue的基本概念

队列(Queue)则是一种先进先出(FIFO)的数据结构

  • 它在处理需要按顺序处理的任务时非常有用。无论是操作系统中的任务调度,还是网络中的数据包传输,队列都扮演着不可或缺的角色。

在这里插入图片描述
在这里插入图片描述


📙2. stack与queue的常用操作

⛰️stack的常用操作

函数说明接口说明
stack()构造空的栈
empty()检测stack是否为空
size()返回stack中元素的个数
top()返回栈顶元素
push()将元素val压入stack中
pop()将stack中尾部的元素弹出
void test_stack()
{stack<int, vector<int>> st; // 构造栈st.push(1); // 将元素val压入stack中st.push(2);st.push(3);st.push(4);printf("栈中有效元素个数为:%d", st.size()); // 返回stack中元素的个数while (!st.empty()) // 检测stack是否为空{printf("%d ", st.top()); // 返回栈顶元素的引用st.pop(); // 将stack中尾部的元素弹出}
}

相较于之前的栈的常用函数学习还是很简单的,在了解完基本用法后,这里推荐几个相关题目


🌄queue的常用操作

函数声明接口说明
queue()构造空的队列
empty()检测队列是否为空,是返回true,否则返回false
size()返回队列中有效元素的个数
front()返回队头元素
back()返回队尾元素
push()在队尾将元素val入队列
pop()将队头元素出队列
void test_queue()
{queue<int, list<int>> q; // 构造队列q.push(1); // 在队尾将元素val入队列q.push(2);q.push(3);q.push(4);printf("队列中有效元素个数为:%d", q.size()); // 返回stack中元素的个数while (!q.empty()) // 检测queue是否为空{printf("%d ", q.front()); // 返回队头元素q.pop(); // 将队头元素出队列}
}

相较于之前的栈的常用函数学习还是很简单的,在了解完基本用法后,这里推荐几个相关题目,不多说直接上题目巩固
最小栈
栈的压入、弹出序列
逆波兰表达式求值
用栈实现队列
用队列实现栈


📚3. 容器适配器

容器适配器是一种机制,它接受一种已有的容器类型,通过封装和改造,使其行为看起来像另一种类型。这允许我们使用特定的数据访问和操作模式(如栈、队列或优先队列)来管理容器中的数据,而无需修改原始容器的实现。

C++标准库定义了三种序列容器适配器:

容器适配器概念
stack(栈)栈是一种后进先出(LIFO)的数据结构,具有push(压栈)、pop(弹栈)、top(查看栈顶元素)等基本操作。在STL中,stack可以建立在vector、list、deque等容器之上。
queue(队列)队列是一种先进先出(FIFO)的数据结构,具有push(入队)、pop(出队)、front(查看队首元素)、back(查看队尾元素)等基本操作。queue在STL中也是一个容器适配器。
priority_queue(优先队列)优先队列是一种特殊的队列,其中元素的出队顺序不是按照它们进入队列的顺序,而是根据它们的优先级。priority_queue提供了push(插入元素)、pop(删除最高优先级元素)、top(查看最高优先级元素)等操作。

虽然stack和queue中也可以存放元素,但在STL中并没有将其划分在容器的行列,而是将其称为容器适配器, 这是因为stack和队列只是对其他容器的接口进行了包装,STL中stack和queue默认使用deque,这个在stack和queue的基本概念中可以看到。
在这里插入图片描述
在这里插入图片描述

有了容器适配器后,我们在模拟实现时,就不需要自己再从底层出发,而是可以直接调用已经存在的容器


📒4. deque的简单介绍

💧deque的原理介绍

deque(双端队列):是一种双开口的"连续"空间的数据结构

  • 双开口的含义是:可以在头尾两端进行插入和删除操作,且时间复杂度为O(1),与vector比较,头插效率高,不需要搬移元素;与list比较,空间利用率比较高

在这里插入图片描述
在这里插入图片描述

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

在这里插入图片描述

我们查表发现deque基本上包含了vector与list的用法,那我们之前为什么还要费尽心思去学习呢?直接学习deque不好吗?


🌊deque的缺陷

与vector比较

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

与list比较

  • 其底层是连续空间,空间利用率比较高,不需要存储额外字段。但是,deque有一个致命缺陷:不适合遍历,因为在遍历时,deque的迭代器要频繁的去检测其是否移动到某段小空间的边界,导致效率低下,

在序列式场景中,可能需要经常遍历,因此在实际中,需要线性结构时,大多数情况下优先考虑vector和list,deque的应用并不多,而目前能看到的一个应用就是,STL用其作为stack和queue的底层数据结构

因此我们还是要单独学习list和vector的


📜5. stack和queue的模拟实现

在模拟实现这俩个容器的时候,我们只要实现它的常用函数,在模拟实现时,我们只需要复用就可以了

🍂stack的模拟实现

template<class T, class Container = deque<T>>
class stack
{
public:void push(const T& x){_con.push_back(x);}void pop(){_con.pop_back();}const T& top(){return _con.back();}bool empty(){return _con.empty();}size_t size(){_con.size();}
private:Container _con;
};

🍁queue的模拟实现

template<class T, class Container = deque<T>>
class queue
{
public:void push(const T& x){_con.push_back(x);}void pop(){_con.pop_front();}bool empty(){return _con.empty();}const T& front(){return _con.front();}size_t size(){_con.size();}
private:Container _con;
};

关于stack与queue的模拟实现就是复用之前学过的函数,没什么好说的,让我们进入重头戏


📖6. priority_queue

🌞priority_queue的基本概念

关于优先级队列,我们就可以把它想象成堆那样的结构
在这里插入图片描述

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

注意:默认情况下priority_queue是大堆


🌙priority_queue的常用操作

函数声明接口说明
priority_queue()/priority_queue(first,last)构造一个空的优先级队列
empty( )检测优先级队列是否为空,是返回true,否则返回false
top( )返回优先级队列中最大(最小元素),即堆顶元素
push(x)在优先级队列中插入元素x
pop()删除优先级队列中最大(最小)元素,即堆顶元素
void test_priority_queue()
{std::priority_queue<int> q; // 构造优先级队列q.push(1); // 在优先级队列中插入元素xq.push(43);q.push(233);q.push(43);q.push(4);q.push(73);q.push(2);while (!q.empty()) // 检测优先级队列是否为空{printf("%d ", q.top()); // 返回堆顶元素q.pop(); // 删除堆顶元素}
}

⭐priority_queue的模拟实现

// 仿函数,数据的比较方法
template<class T>
class Less
{
public:bool operator()(const T& x, const T& y){return x < y;}
};
template<class T>
class Greater
{
public: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
{
public:priority_queue() // 无参构造{}template<class InputIterator>priority_queue(InputIterator first, InputIterator last) // 迭代器构造:_con(first,last){// 向下调整建堆for (int i = (_con.size() - 2) / 2; i >= 0; i--){AdjustDown(i);}}void AdjustUp(int child) // 向上调整{int parent = (child - 1) / 2;Compare com;while (child > 0){if (com(_con[child], _con[parent])){swap(_con[child], _con[parent]);child = parent;parent = (child - 1) / 2;}else{break;}}}void AdjustDown(int parent) // 向下调整{int child = parent * 2 + 1;Compare com;while (com(child , _con.size())){if ((child + 1 < _con.size()) &&com(_con[child + 1], _con[child])){child++;}if (com(_con[child] , _con[parent])){swap(_con[child], _con[parent]);parent = child;child = parent * 2 + 1;}else{break;}}}void push(const T& x){_con.push_back(x);AdjustUp(_con.size() - 1); // 在插入新元素时,我们要调整容器里面的元素}size_t size(){return _con.size();}const T& top(){return _con[0];}void pop(){swap(_con[0], _con[_con.size() - 1]);_con.pop_back(); AdjustDown(0); // 删除时,依旧要调整元素}bool empty(){return _con.empty();}
private:Container _con;
};

关于priority_queue的模拟实现,依然是以复用为主,只不过多了一些堆要使用的调整函数,而且我们在查priority_queue这个容器时,不难发现其实它是有三个模板参数的,它的第三个模板参数就是,数据的排列方式,也就是决定大小堆的,这就涉及到了仿函数,关于仿函数,下节我们在讲!


🔥7. 总结

在深入探讨了STL(Standard Template Library)中的栈(stack)与队列(queue)之后,我们不难发现这两大数据结构在程序设计中扮演着至关重要的角色。栈与队列的引入,不仅极大地丰富了C++程序员的工具箱,也为解决各种实际问题提供了更为高效、优雅的方法

学习栈与队列并不仅仅是为了掌握它们的基本操作和应用场景。更重要的是,我们要学会如何根据问题的特点选择合适的数据结构,以及如何有效地利用数据结构来解决实际问题。在这个过程中,我们需要不断思考、探索和实践,才能不断提升自己的编程能力和问题解决能力。让我们继续前行,不断探索编程的奥秘,享受编程的乐趣!

在这里插入图片描述

谢谢大家支持本篇到这里就结束了,端午安康,祝大家天天开心!
在这里插入图片描述

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

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

相关文章

探索软件工程师在新能源汽车研发中的角色与贡献

随着全球对可持续发展的关注不断增加&#xff0c;新能源汽车的研发与应用成为了汽车行业的一个重要方向。作为软件工程师&#xff0c;参与新能源汽车研发不仅能够推动科技创新&#xff0c;还能为环保事业贡献力量。本文将深入探讨软件工程师在新能源汽车研发中的具体贡献、所需…

C#操作MySQL从入门到精通(20)——更新数据

前言: 谈到数据库,大家最容易脱口而出的就是增删改查,本文所说的更新数据就是增删改查的改,改变数据的意思。 本文测试使用的数据库如下: 1、更新一列 所谓更新一列的意思就是只更改一列数据,并且通常要使用where条件,因为不加这个条件的话会导致将所有行的数据进行…

数据可视化Python实现超详解【数据分析】

各位大佬好 &#xff0c;这里是阿川的博客&#xff0c;祝您变得更强 个人主页&#xff1a;在线OJ的阿川 大佬的支持和鼓励&#xff0c;将是我成长路上最大的动力 阿川水平有限&#xff0c;如有错误&#xff0c;欢迎大佬指正 Python 初阶 Python–语言基础与由来介绍 Python–…

后端服务器启动访问

VisualStudioCode(VSCode) 服务器启动 浏览器中测试访问 http://localhost:3000

算法笔记1-高精度模板(加减乘除)个人模板

目录 加法 减法 乘法 ​编辑 除法 加法 #include <iostream> #include <cstring> #include <algorithm> #include <cmath> #include <queue>using namespace std;typedef pair<int,int> PII;const int N 1e5 10;int n; int a[N],…

ROS学习记录:C++节点发布自定义地图

前言 ROS栅格地图格式 在了解了ROS地图消息包的数据结构后(链接在上)&#xff0c;本文将编写一个节点&#xff0c;发布地图消息包&#xff0c;看看在RViz中显示是什么效果。 一、准备 1、为了简单起见&#xff0c;发布一个两行四列的地图 2、为了便于观测&#xff0c;只对地…

SmartEDA VS Multisim/Proteus:电子设计江湖,谁主沉浮?

在电子设计的江湖里&#xff0c;SmartEDA、Multisim和Proteus无疑是几大门派&#xff0c;各自拥有独特的武功秘籍和门派特色。今天&#xff0c;我们就来一场巅峰对决&#xff0c;看看这些电子设计软件究竟谁能笑傲江湖&#xff0c;成为电子设计界的霸主&#xff01; 一、门派起…

Seq2seq、编码器解码器神经网络

目录 一、Seq2seq 简介二、编码器三、解码器四、编码器-解码器的训练 遇到看不明白的地方&#xff0c;欢迎在评论中留言呐&#xff0c;一起讨论&#xff0c;一起进步&#xff01; 需掌握的前提知识&#xff1a; LSTM、词嵌入 本文参考&#xff1a;【官方双语】编码、解码神经网…

tkinter菜单栏

tkinter菜单栏 菜单栏效果代码 菜单栏 在 Tkinter 中&#xff0c;Menu 组件用于创建菜单栏、下拉菜单和上下文菜单&#xff0c;是构建图形用户界面&#xff08;GUI&#xff09;应用程序的常见需求。 效果 代码 import tkinter as tk from tkinter import messagebox# 创建主…

DAMA学习笔记(一)-数据管理

1.引言 数据管理(Data Management) 是为了 交付、 控制、 保护 并 提升 数据和信息资产的 价值 , 在其整个生命周期中制订 计划、 制度、 规程和实践 活动, 并 执行 和 监督 的过程。 数据管理专业人员(Data Management Professional) 是指 从事数据管理各方面的工作…

MySQL与PostgreSQL关键对比三(索引类型)

目录 索引类型 B-tree 索引 Hash 索引 Full-text 索引 GiST 索引 GIN 索引 BRIN 索引 索引创建示例 MySQL PostgreSQL 结论 以下SQL语句的执行如果需要开发工具支持&#xff0c;可以尝试使用SQLynx或Navicat来执行。 MySQL和PostgreSQL在索引方面有许多相似之处&am…

【C#线程设计】2:backgroundWorker

实现&#xff1a; &#xff08;1&#xff09;.控件&#xff1a;group Box&#xff0c;text Box&#xff0c;check Box&#xff0c;label&#xff0c;botton&#xff0c;richtextbox 控件拉取见&#xff1a;https://blog.csdn.net/m0_74749240/article/details/139409510?spm1…

吴恩达2022机器学习专项课程C2W3:2.25 理解方差和偏差(诊断方差偏差正则化偏差方案)

目录 引言名词替代影响模型偏差和方差的因素1.多项式阶数2.正则化参数 判断是否有高偏差或高方差1.方法一&#xff1a;建立性能基准水平2.方法二&#xff1a;建立学习曲线 总结 引言 机器学习系统开发的典型流程是从一个想法开始&#xff0c;然后训练模型。初次训练的结果通常…

C语言最终讲:预处理详解

C语言最终讲&#xff1a;预处理详解 1.预定义符号2.#define定义常量3.#define定义宏4.带有副作用的宏参数5.宏替换的规则6.宏和函数的对比6.1宏的优势6.1.1\符号 6.2宏的劣势 7.#和##7.1#运算符7.2##运算符 8.命名约定9.#undef10.命令行定义11.条件编译12.头文件的包含12.1本地…

13. UDP协议与RTP协议

UDP协议 UDP协议比较简单&#xff1a; UDP的长度是固定的&#xff0c;用总长度-UDP长度就是数据长度。 UDP是不保证他的有序性和可靠性的。对于音频和视频是这样是比较好的&#xff0c;因为这段丢了&#xff0c;我们可以从下一段在开始解码。 RTP RTP 协议概述 RTP&#x…

【MySQL】(基础篇六) —— 过滤数据

过滤数据 本文将讲授如何使用SELECT语句的WHERE子句指定搜索条件。 WHERE子句 数据库表一般包含大量的数据&#xff0c;很少需要检索表中所有行。通常只会根据特定操作或需要提取表数据的子集。只检索所需数据需要指定搜索条件&#xff08;search criteria&#xff09;&…

代码随想录算法训练营第36期DAY56

DAY56 套磁很顺利&#xff0c;发现又有书读了&#xff01; 300最长递增子序列 朴素法&#xff0c;这个好想&#xff0c;但是不对&#xff0c;比如 0 1 0 3 2 3 我的算法会找出0 1 3作为答案&#xff0c;而不是0 1 2 3 可以看出&#xff0c;后面的状态依赖于前面的状态&am…

Facebook革新:数字社交的下一个阶段

在数字化时代&#xff0c;社交网络已经成为人们生活中不可或缺的一部分。作为全球最大的社交网络平台之一&#xff0c;Facebook一直在不断创新&#xff0c;引领着数字社交的发展。然而&#xff0c;随着科技的不断进步和社交需求的变化&#xff0c;Facebook正在走向一个新的阶段…

Gitte的使用(Windows/Linux)

Gitte的使用&#xff08;Windows/Linux&#xff09; 一、Windows上使用Gitte1.下载程序2.在Gitte上创建远程仓库3.连接远程仓库4.推送文件到远程仓库 二、Linux上使用Gitte1.第一次从仓库上传1.1生成公钥1.2配置SSH公钥1.3新建一个仓库1.4配置用户名和邮箱在Linux中1.5创建仓库…

python字典应用

""" 字典应用 字典中保存了股票信息&#xff0c;完成下面的操作 1.找出股票价格大于100元的股票并创建一个新的字典 2、找出价格最高和最低的股票对应的股票代码 3.按照股票价格从高到低给股票代码排序 """stocks {AAPL: 191.88,G00G: 1186.96,…