C++:栈(stack)、队列(queue)、优先级队列(priority_queue)

2024-06-04 19:52

本文主要是介绍C++:栈(stack)、队列(queue)、优先级队列(priority_queue),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

hello,各位小伙伴,本篇文章跟大家一起学习《C++:栈(stack)和队列(queue)》,感谢大家对我上一篇的支持,如有什么问题,还请多多指教 !

文章目录

    • :maple_leaf:栈---stack
    • :maple_leaf:栈---stack题目:最小栈(来自leetcode)
      • :leaves:如果途中出栈
      • :leaves:答案代码
    • :maple_leaf:栈的压入、弹出序列
      • :leaves:答案代码
    • :maple_leaf:队列---queue
    • :maple_leaf:优先级队列---priority_queue
      • :leaves:优先级队列使用
      • :leaves:注意

🍁栈—stack

template <class T, class Container = deque > class stack;
栈是一种容器适配器,专门设计用于在后进先出(后进先出)上下文中操作,其中元素仅从容器的一端插入和提取。

栈—stack被实现为容器适配器,它是使用特定容器类的封装对象作为其底层容器的类,提供一组特定的成员函数来访问其元素。元素是从特定容器的“后部”推/弹出的,即堆栈的顶部。底层容器可以是任何标准容器类模板或某些其他专门设计的容器类。

容器应支持以下操作:

  • empty
  • size
  • back
  • push_back
  • pop_back

标准容器类vector、deque和list满足这些要求。默认情况下,如果没有为特定堆栈类实例化指定容器类,则使用标准容器deque

是后进先出的底层容器,是一个模板类,其逻辑结构:

在这里插入图片描述

栈的成员函数以及功能:
在这里插入图片描述

🍁栈—stack题目:最小栈(来自leetcode)

题目在这:最小栈

在这里插入图片描述
题目会对该栈进行压栈和出栈,要我们随时查找栈里最小的元素,并且规定在常数时间里检索出来并返回。

根据栈的性质—LIFO(后进先出):
设计两个栈

  • _push负责接受所有操作(1.压栈、2.出栈)
  • s另一个负责接收最小元素(1.当s为空或者栈顶元素 >= _push压栈元素时压栈,2.当_push出栈元素 == s栈顶元素时出栈,3.获取栈顶元素)
  • 测试例子

在这里插入图片描述

第一步操作:对_push栈进行压栈(元素为-2),由于一开始s栈为空,所以s进行压栈:
在这里插入图片描述
第二步操作:对_push栈进行压栈(元素为0),由于0 > s的栈顶元素 -2,所以s栈不进行操作:
在这里插入图片描述
第三步操作:对_push栈进行压栈(元素为-3),由于-3 < s的栈顶元素 -2,所以s栈进行压栈操作:
在这里插入图片描述
第四步操作:minStack.getMin();
返回s栈的栈顶元素 -> -3

🍃如果途中出栈

上述例子,在第四步操作minStack.getMin();_push出栈一次,由于_push出栈元素等于s栈顶元素,所以s也跟着出栈,最后返回结果是-2
在这里插入图片描述

🍃答案代码

class MinStack {
public:MinStack() {}void push(int val) {s.push(val);if(_push.empty() || _push.top()>=val){_push.push(val);}}void pop() {int tmp = s.top();s.pop();if(_push.top() == tmp){_push.pop();}}int top() {return s.top();}int getMin() {return _push.top();}stack<int> _push;stack<int> s;
};

通过:
在这里插入图片描述

🍁栈的压入、弹出序列

题目在这:栈的压入、弹出序列
在这里插入图片描述
题目给出两个序列vector<int>& pushV, vector<int>& popV,第一个序列表示栈的压入顺序,请判断第二个序列是否可能为该栈的弹出顺序。

注意:题目说明压栈的所有数字都不相等!
那么我们可以设计一个栈_push,通过模拟出栈来解决这个问题。

int popi = 0;
int pushi = 0;
int len = popV.size();
if(len == 0)// 当队列为空时,返回true
return true;

遍历两个队列vector<int>& pushV, vector<int>& popV

  1. pushV[pushi] != popV[popi]时,那么说明并没有出栈,只有压栈,那么_push进行压栈操作:
    while循环来控制
while(pushi < len)
if(pushV[pushi] != popV[popi])
{_push.push(pushV[pushi]);++pushi;
}
  1. pushV[pushi] == popV[popi]时,那么说明并存在出栈,先压栈后出栈,那么_push进行压栈出栈操作,出栈结束后还要判断_push栈顶元素是否等于popV[popi],如果相等则继续出栈(注意,此时_push不能为空,_push为空退出判断,popi > len说明出栈任务结束,退出判断):
_push.push(pushV[pushi]);// 先压栈
++pushi;
while(!_push.empty() && popi < len 
&& _push.top() == popV[popi])
{_push.pop();++popi;
}
  1. while(pushi < len)结束,最后进行判断第二个序列是否可能为该栈的弹出顺序。很简单,判断_push是否为空或者popi 是否等于len即可。

🍃答案代码

class Solution {
public:/*** 代码中的类名、方法名、参数名已经指定,请勿修改,直接返回方法规定的值即可** * @param pushV int整型vector * @param popV int整型vector * @return bool布尔型*/bool IsPopOrder(vector<int>& pushV, vector<int>& popV) {// write code hereint popi = 0;int pushi = 0;int len = popV.size();if(len == 0)return true;while(pushi < len){if(pushV[pushi] != popV[popi]){_push.push(pushV[pushi]);++pushi;}else{_push.push(pushV[pushi]);++pushi;while(!_push.empty() && popi < len && _push.top() == popV[popi]){_push.pop();++popi;}}}return popi == len;// return _push.empty();}stack<int> _push;
};

在这里插入图片描述

🍁队列—queue

template <class T, class Container = deque > class queue;
队列是一种容器适配器,专门设计用于在 FIFO 上下文(先进先出)中操作,其中元素被插入到容器的一端并从另一端提取。

队列被实现为容器适配器,它们是使用特定容器类的封装对象作为其底层容器的类,提供一组特定的成员函数来访问其元素。元素被推入特定容器的后面并从其前面弹出。

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

  • empty
  • size
  • front
  • back
  • push_back
  • pop_front

标准容器类 deque 和 list 满足这些要求。默认情况下,如果没有为特定队列类实例化指定容器类,则使用标准容器双端队列。

  • 逻辑图
    在这里插入图片描述
    队列的成员函数及其功能
    在这里插入图片描述

🍁优先级队列—priority_queue

  1. 优先队列是一种容器适配器,根据严格的弱排序标准,它的第一个元素总是它所包含的元素中最大的。
  2. 此上下文类似于堆,在堆中可以随时插入元素,并且只能检索最大堆元素(优先队列中位于顶部的元素)。
  3. 优先队列被实现为容器适配器,容器适配器即将特定容器类封装作为其底层容器类,queue提供一组特定的成员函数来访问其元素。元素从特定容器的尾部弹出,其称为优先队列的顶部。
  4. 底层容器可以是任何标准容器类模板,也可以是其他特定设计的容器类。容器应该可以通过随机访问迭
    代器访问,并支持以下操作:
  • empty():检测容器是否为空
  • size():返回容器中有效元素个数
  • front():返回容器中第一个元素的引用
  • push_back():在容器尾部插入元素
  1. 标准容器类vectordeque满足这些需求。默认情况下,如果没有为特定的priority_queue类实例化指定容器类,则使用vector
  2. 需要支持随机访问迭代器,以便始终在内部保持堆结构。容器适配器通过在需要时自动调用算法函数make_heap、push_heappop_heap来自动完成此操作。
  • 简而言之就是堆:
    优先级队列默认使用vector作为其底层存储数据的容器,在vector上又使用了堆算法将vector中元素构造成堆的结构,因此priority_queue就是堆,所有需要用到堆的位置,都可以考虑使用priority_queue。注意:默认情况下priority_queue是大堆。

🍃优先级队列使用

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

在这里插入图片描述

🍃注意

  • priority_queue第三个参数有缺省值less,所以priority_queue默认是大堆,如果要建小堆,将第三个模板参数换成greater比较方式,如:
vector<int> v{3,2,7,6,0,4,1,9,8,5};
priority_queue<int, vector<int>, greater<int>> q2(v.begin(), v.end());
cout << q2.top() << endl;
  • 如果在priority_queue中放自定义类型的数据,用户需要在自定义类型中提供 > 或者 < 的重载。
    像是Date日期类的比较(年、月、日)。

你学会了吗?
好啦,本章对于《C++:栈(stack)和队列(queue)》的学习就先到这里,如果有什么问题,还请指教指教,希望本篇文章能够对你有所帮助,我们下一篇见!!!

如你喜欢,点点赞就是对我的支持,感谢感谢!!!

请添加图片描述

这篇关于C++:栈(stack)、队列(queue)、优先级队列(priority_queue)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1030961

相关文章

c++中std::placeholders的使用方法

《c++中std::placeholders的使用方法》std::placeholders是C++标准库中的一个工具,用于在函数对象绑定时创建占位符,本文就来详细的介绍一下,具有一定的参考价值,感兴... 目录1. 基本概念2. 使用场景3. 示例示例 1:部分参数绑定示例 2:参数重排序4. 注意事项5.

使用C++将处理后的信号保存为PNG和TIFF格式

《使用C++将处理后的信号保存为PNG和TIFF格式》在信号处理领域,我们常常需要将处理结果以图像的形式保存下来,方便后续分析和展示,C++提供了多种库来处理图像数据,本文将介绍如何使用stb_ima... 目录1. PNG格式保存使用stb_imagephp_write库1.1 安装和包含库1.2 代码解

C++实现封装的顺序表的操作与实践

《C++实现封装的顺序表的操作与实践》在程序设计中,顺序表是一种常见的线性数据结构,通常用于存储具有固定顺序的元素,与链表不同,顺序表中的元素是连续存储的,因此访问速度较快,但插入和删除操作的效率可能... 目录一、顺序表的基本概念二、顺序表类的设计1. 顺序表类的成员变量2. 构造函数和析构函数三、顺序表

使用C++实现单链表的操作与实践

《使用C++实现单链表的操作与实践》在程序设计中,链表是一种常见的数据结构,特别是在动态数据管理、频繁插入和删除元素的场景中,链表相比于数组,具有更高的灵活性和高效性,尤其是在需要频繁修改数据结构的应... 目录一、单链表的基本概念二、单链表类的设计1. 节点的定义2. 链表的类定义三、单链表的操作实现四、

Linux之进程状态&&进程优先级详解

《Linux之进程状态&&进程优先级详解》文章介绍了操作系统中进程的状态,包括运行状态、阻塞状态和挂起状态,并详细解释了Linux下进程的具体状态及其管理,此外,文章还讨论了进程的优先级、查看和修改进... 目录一、操作系统的进程状态1.1运行状态1.2阻塞状态1.3挂起二、linux下具体的状态三、进程的

解读Redis秒杀优化方案(阻塞队列+基于Stream流的消息队列)

《解读Redis秒杀优化方案(阻塞队列+基于Stream流的消息队列)》该文章介绍了使用Redis的阻塞队列和Stream流的消息队列来优化秒杀系统的方案,通过将秒杀流程拆分为两条流水线,使用Redi... 目录Redis秒杀优化方案(阻塞队列+Stream流的消息队列)什么是消息队列?消费者组的工作方式每

使用C/C++调用libcurl调试消息的方式

《使用C/C++调用libcurl调试消息的方式》在使用C/C++调用libcurl进行HTTP请求时,有时我们需要查看请求的/应答消息的内容(包括请求头和请求体)以方便调试,libcurl提供了多种... 目录1. libcurl 调试工具简介2. 输出请求消息使用 CURLOPT_VERBOSE使用 C

C++实现获取本机MAC地址与IP地址

《C++实现获取本机MAC地址与IP地址》这篇文章主要为大家详细介绍了C++实现获取本机MAC地址与IP地址的两种方式,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 实际工作中,项目上常常需要获取本机的IP地址和MAC地址,在此使用两种方案获取1.MFC中获取IP和MAC地址获取

C/C++通过IP获取局域网网卡MAC地址

《C/C++通过IP获取局域网网卡MAC地址》这篇文章主要为大家详细介绍了C++如何通过Win32API函数SendARP从IP地址获取局域网内网卡的MAC地址,感兴趣的小伙伴可以跟随小编一起学习一下... C/C++通过IP获取局域网网卡MAC地址通过win32 SendARP获取MAC地址代码#i

C++中使用vector存储并遍历数据的基本步骤

《C++中使用vector存储并遍历数据的基本步骤》C++标准模板库(STL)提供了多种容器类型,包括顺序容器、关联容器、无序关联容器和容器适配器,每种容器都有其特定的用途和特性,:本文主要介绍C... 目录(1)容器及简要描述‌php顺序容器‌‌关联容器‌‌无序关联容器‌(基于哈希表):‌容器适配器‌:(