容器适配器-stack栈

C++标准库不只是包含了顺序容器,还包含一些为满足特殊需求而设计的容器,它们提供简单的接口。

这些容器可被归类为容器适配器(container adapter),它们是改造别的标准顺序容器,使之满足特殊需求的新容器。

适配器:也称配置器,把一种接口转为另一种接口。

有三种标准的 容器适配器: stack(栈)、queue(队列)和priority queue(优先级队列)。 priority queue就是“根据排序准则,自动将元素排序”的队列,其所谓“下一个””元素总是拥有最高优先级。

stack栈

stack 栈是一种只在一端(栈顶)进行数据插入(入栈)和删除(出栈)的数据结构,它满足后进先出(LIFO)的特性。

使用push(入栈)将数据放入stack,使用pop(出栈)将元素从容器中移除。 POP PUSH栈顶TOP栈底

 

使用stack,必须包含头文件:

#include <stack>

在头文件中,class stack定义如下:

namespace std{template <typename T,typename Container = deque<T>>class stack;
}

第一个参数T代表类型,第二个参数用来定义stack内部存放数据的容器,默认为deque。之所以选择deque而非vector,是因为deque移除数据时可能会释放内存,并在插入数据需要扩容时不需要复制所有的数据。

例如,以下定义了一个元素类型为整数的 stack:

std::stack<int> st;

stack 只是很单纯地把各项操作转化为内部容器对应的函数调用。你可以使用任何支持 back()、push_back()和pop_back()成员函数的标准容器支持 stack。例如你可以使用 vector或list 来存放数据:

stack<int,vector<int>> st;//整型栈,使用vector存放数据

注意:forword_list和array不可以作为其容器

定义及初始化

#include <iostream>
#include <stack>
#include <vector>
#include <list>
using namespace std;int main()
{stack <char> s1;//创建一个默认的栈,最常用stack <char, deque<char> > s2;//显示创建用deque保存数据的栈,和s1等价stack <int, vector<int> > s3;//创建用vector保存数据的栈stack <int, list<int> > s4;//创建用list保存数据的栈return 0;
}

先创建一个空的栈,然后往里面存放数据。

常用操作

stack栈的操作比较简单,不支持迭代器和运算符,只有几个简单的成员函数。 empty成员函数

empty成员函数

判断栈是否为空。

//判空
bool IsEmpty(PLStack ps)
{return ps->next == NULL;
}

pop成员函数

出栈函数,删除栈顶元素。

//获取栈顶元素的值,并且删除
bool Pop(PLStack ps, int* rtval)
{assert(ps != NULL);if (ps == NULL)return false;if (IsEmpty(ps))return false;*rtval = ps->next->date;LSNode* p = ps->next;ps->next = ps->next->next;free(p);return true;
}

push成员函数

入栈函数,往栈顶添加数据。

//往栈中入数据(入栈操作)
bool Push(PLStack ps, int val)
{assert(ps != NULL);if (ps == NULL)return false;LSNode* p = (LSNode*)malloc(sizeof(LSNode));assert(p != NULL);p->date = val;p->next = ps->next;ps->next = p;return false;
}

size成员函数

返回栈的数据个数。

//获取栈中有效数据的个数
int GetLength(PLStack ps)
{assert(ps != NULL);if (ps == NULL)return 0;int count = 0;for (LSNode* p = ps->next; p != NULL; p = p->next){count++;}return count;
}

top成员函数

返回栈顶元素的引用。

//获取栈顶元素的值,但不删除
bool  GetTop(PLStack ps, int* rtval)
{assert(ps != NULL);if (ps == NULL)return false;if (IsEmpty(ps))return false;*rtval = ps->next->date;return true;
}

栈的实现方式​

链表实现:链表实现栈则更加灵活,它不需要预先分配固定大小的内存空间,适用于栈的大小动态变化且难以预估的场景。在链表实现中,每个节点包含数据和指向下一个节点的指针。栈顶对应链表的头节点,入栈和出栈操作都在链表头部进行,时间复杂度同样为 O (1)。

.h文件

#pragma once//链式栈
typedef struct LSNode
{int date;struct LSNode* next;
}LSNode, * PLStack;//初始化
void InitStack(PLStack ps);//往栈中入数据(入栈操作)
bool Push(PLStack ps, int val);//获取栈顶元素的值,但不删除
bool  GetTop(PLStack ps, int* rtval);//获取栈顶元素的值,并且删除
bool Pop(PLStack ps, int* rtval);//判空
bool IsEmpty(PLStack ps);//获取栈中有效数据的个数
int GetLength(PLStack ps);//清空所有的数据
void Clear(PLStack ps);//销毁
void Destroy(PLStack ps);

.cpp文件

#include "Istak.h"
#include <iostream>
#include <cassert>
using namespace std;//初始化
void InitStack(PLStack ps)
{assert(ps != NULL);if (ps == NULL)return;ps->next = NULL;
}//往栈中入数据(入栈操作)
bool Push(PLStack ps, int val)
{assert(ps != NULL);if (ps == NULL)return false;LSNode* p = (LSNode*)malloc(sizeof(LSNode));assert(p != NULL);p->date = val;p->next = ps->next;ps->next = p;return false;
}//获取栈顶元素的值,但不删除
bool  GetTop(PLStack ps, int* rtval)
{assert(ps != NULL);if (ps == NULL)return false;if (IsEmpty(ps))return false;*rtval = ps->next->date;return true;
}//获取栈顶元素的值,并且删除
bool Pop(PLStack ps, int* rtval)
{assert(ps != NULL);if (ps == NULL)return false;if (IsEmpty(ps))return false;*rtval = ps->next->date;LSNode* p = ps->next;ps->next = ps->next->next;free(p);return true;
}//判空
bool IsEmpty(PLStack ps)
{return ps->next == NULL;
}//获取栈中有效数据的个数
int GetLength(PLStack ps)
{assert(ps != NULL);if (ps == NULL)return 0;int count = 0;for (LSNode* p = ps->next; p != NULL; p = p->next){count++;}return count;
}//清空所有的数据
void Clear(PLStack ps)
{Destroy(ps);
}//销毁
void Destroy(PLStack ps)
{LSNode* p;while (ps->next != NULL){p = ps->next;ps->next = p->next;free(p);}
}

栈的应用场景​

1.表达式求值

在数学表达式求值中,栈发挥着关键作用。例如,对于后缀表达式(逆波兰表达式),我们可以利用栈来高效地计算结果。后缀表达式将运算符放在操作数之后,避免了括号带来的优先级判断复杂性。计算时,从左到右扫描后缀表达式,遇到操作数就将其入栈,遇到运算符则从栈中弹出相应数量的操作数进行运算,并将结果入栈。最终,栈顶元素即为表达式的计算结果。以表达式 “3 4 + 2 *” 为例,计算过程如下:​

扫描到 3,入栈,栈:[3]​

扫描到 4,入栈,栈:[3, 4]​

扫描到 +,弹出 4 和 3,计算 3 + 4 = 7,将 7 入栈,栈:[7]​

扫描到 2,入栈,栈:[7, 2]​

扫描到 *,弹出 2 和 7,计算 7 * 2 = 14,将 14 入栈,栈:[14]​ 

最终结果为 14

2.括号匹配

在编译器和文本编辑器中,常常需要检查代码中的括号是否正确匹配,如圆括号、方括号和花括号。利用栈可以轻松解决这个问题。当遇到左括号时,将其入栈;遇到右括号时,从栈中弹出对应的左括号进行匹配。如果在匹配过程中栈为空或者匹配失败,说明括号不匹配。例如,对于字符串 “{[()]}”,处理过程如下:​

遇到 {,入栈,栈:[{]​

遇到 [,入栈,栈:[{, []​

遇到 (,入栈,栈:[{, [, (]​

遇到 ),弹出 (进行匹配,栈:[{, []​

遇到 ],弹出 [进行匹配,栈:[{]​

遇到 },弹出 {进行匹配,栈:[]​

匹配成功

3.函数调用栈

在程序执行过程中,函数调用是通过栈来管理的。当一个函数被调用时,它的相关信息,如参数、局部变量和返回地址等,会被压入栈中。当函数执行完毕返回时,这些信息会从栈中弹出,程序继续执行调用函数的下一条指令。例如,在一个包含多个函数嵌套调用的程序中,栈记录了函数调用的顺序和状态,确保函数能够正确返回和继续执行。假设函数 A 调用函数 B,函数 B 又调用函数 C,栈的状态变化如下:​

调用 A,A 的相关信息入栈,栈:[A]​

A 中调用 B,B 的相关信息入栈,栈:[A, B]​

B 中调用 C,C 的相关信息入栈,栈:[A, B, C]​

C 执行完毕返回,C 的信息出栈,栈:[A, B]​

B 执行完毕返回,B 的信息出栈,栈:[A]​

A 执行完毕返回,A 的信息出栈,栈:[]

4.深度优先搜索(DFS)

在图论和搜索算法中,深度优先搜索常常用栈来实现。在遍历图的过程中,从起始节点开始,将其入栈。然后,每次从栈中弹出一个节点,访问该节点,并将其未访问过的邻接节点入栈,直到栈为空。这样的方式使得搜索沿着一条路径尽可能深地探索下去,直到无法继续,然后回溯到上一个节点继续探索其他路径。

5.浏览器历史记录

我们日常使用的浏览器,其历史记录功能也运用了栈的思想。当我们依次访问网页 A、B、C 时,这些网页依次被压入栈中。当我们点击 “后退” 按钮时,就相当于执行出栈操作,从栈顶弹出当前页面,回到上一个页面。例如,访问顺序为 A -> B -> C,栈的状态为 [C, B, A],点击 “后退”,C 出栈,回到 B 页面,栈变为 [B, A]

总结​

栈作为一种简单而强大的数据结构,以其独特的后进先出特性,在众多领域展现出高效的应用价值。无论是复杂的算法实现,还是日常使用的软件功能,栈都默默发挥着重要作用。通过对栈的概念、操作、实现方式以及应用场景的深入学习,我们不仅掌握了一种基础的数据结构,更能在编程实践中灵活运用它来解决各种实际问题。在未来的编程之旅中,希望你能充分利用栈的优势,优化代码,提升程序的性能和效率。让我们继续探索数据结构的奇妙世界,解锁更多编程的奥秘!

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

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

相关文章

[250403] HuggingFace 新增检查模型与电脑兼容性的功能 | Firefox 发布137.0 支持标签组

目录 Hugging Face 让寻找兼容的 AI 模型变得更容易Firefox 137 版本更新摘要 Hugging Face 让寻找兼容的 AI 模型变得更容易 Hugging Face 是一个流行的在线平台&#xff0c;用于访问开源人工智能 (AI) 工具和模型。该平台推出了一项有用的新功能&#xff0c;允许个人轻松检查…

.NET 创建MCP使用大模型对话二:调用远程MCP服务

在上一篇文章.NET 创建MCP使用大模型对话-CSDN博客中&#xff0c;我们简述了如何使用mcp client使用StdIo模式调用本地mcp server。本次实例将会展示如何使用mcp client模式调用远程mcp server。 一&#xff1a;创建mcp server 我们创建一个天气服务。 新建WebApi项目&#x…

Redis 中 Set(例如标签) 和 ZSet(例如排行榜) 的详细对比,涵盖定义、特性、命令、适用场景及总结表格

以下是 Redis 中 Set 和 ZSet 的详细对比&#xff0c;涵盖定义、特性、命令、适用场景及总结表格&#xff1a; 1. 核心定义 数据类型SetZSet&#xff08;Sorted Set&#xff09;定义无序的、唯一的字符串集合&#xff0c;元素不重复。有序的、唯一的字符串集合&#xff0c;每个…

解决Spring参数解析异常:Name for argument of type XXX not specified

前言 在开发 Spring Boot 应用时&#xff0c;我们常遇到类似 java.lang.IllegalArgumentException: Name for argument not specified 的报错。这类问题通常与方法参数名称的解析机制相关&#xff0c;尤其在使用 RequestParam、PathVariable 等注解时更为常见。 一、问题现象与…

刚刚,OpenAI开源PaperBench,重塑顶级AI Agent评测

今天凌晨1点&#xff0c;OpenAI开源了一个全新的AI Agent评测基准——PaperBench。 这个基准主要考核智能体的搜索、整合、执行等能力&#xff0c;需要对2024年国际机器学习大会上顶尖论文的复现&#xff0c;包括对论文内容的理解、代码编写以及实验执行等方面的能力。 根据O…

Golang封装Consul 服务发现库

以下是一个经过生产验证的 Consul 服务发现封装库,支持注册/注销、健康检查、智能发现等核心功能,可直接集成到项目中: package consulimport ("context""fmt""log""math/rand""net""os""sync"&quo…

自适应信号处理任务(过滤,预测,重建,分类)

自适应滤波 # signals creation: u, v, d N = 5000 n = 10 u = np.sin(np.arange(0, N/10., N/50000

PyTorch深度学习框架 的基础知识

目录 1.pyTorch检查是否安装成功 2.PyTorch的张量tensor 基础创建方式&#xff08;三种&#xff09; 2.2用列表创建tensor 2.2使用元组创建 tensor 2.3使用ndarray创建创建 tensor 2.4 快速创建tensor的常用方法 3.pyTorch中的张量tensor的常用属性 4. tensor中的基础数据…

MySQL学习集--DDL

DDL 数据库操作 查询所有数据库 SHOW DATABASES;查询当前数据库 SELECT DATABASE();创建 CREATE DATABASE[IF NOT EXISTS]数据库名[DEFAULT CHARSET 字符集][COLLATE 排序规则];删除 DROR DATABASE[IF EXISTS]数据库名;使用 USE 数据库名;表操作 创建表格 CREATE TABL…

Vue 3 中按照某个字段将数组分成多个数组

方法一&#xff1a;使用 reduce 方法 const originalArray [{ id: 1, category: A, name: Item 1 },{ id: 2, category: B, name: Item 2 },{ id: 3, category: A, name: Item 3 },{ id: 4, category: C, name: Item 4 },{ id: 5, category: B, name: Item 5 }, ];const grou…

LeetCode刷题 -- 48. 旋转图像

题目 算法题解&#xff1a;顺时针旋转矩阵&#xff08;90度&#xff09; 1. 算法描述 给定一个 n n 的二维矩阵&#xff0c;请将矩阵顺时针旋转 90 度。 例如&#xff1a; 输入&#xff1a; [[1,2,3],[4,5,6],[7,8,9] ]输出&#xff1a; [[7,4,1],[8,5,2],[9,6,3] ]2. 思…

Vulkan进阶系列1 - Vulkan应用程序结构(完整代码)

一: 概述 在前面的20多篇文章中,我们了解了Vulkan的基础知识,和相关API的使用,接下来我们要从零开始写一套完整Vulkan应用程序,在这个过程中加深对Vulkan中的各种概念的理解。 Vulkan 应用程序一般遵循 初始化 -> 运行循环 -> 资源清理 的结构,本实例也基本遵循了…

VTK的两种显示刷新方式

在类中先声明vtk的显示对象 vtkRenderer out_render; vtkVertexGlyphFilter glyphFilter; vtkPolyDataMapper mapper; // 新建制图器 vtkActor actor; // 新建角色 然后在init中先初始化一下&#xff1a; out_rend…

【CSS3】04-标准流 + 浮动 + flex布局

本文介绍浮动与flex布局。 目录 1. 标准流 2. 浮动 2.1 基本使用 特点 脱标 2.2 清除浮动 2.2.1 额外标签法 2.2.2 单伪元素法 2.2.3 双伪元素法(推荐) 2.2.4 overflow(最简单) 3. flex布局 3.1 组成 3.2 主轴与侧轴对齐方式 3.2.1 主轴 3.2.2 侧轴 3.3 修改主…

详细介绍一下C++的按位运算

在C中&#xff0c;按位运算&#xff08;Bitwise Operations&#xff09; 是直接对二进制位&#xff08;bit&#xff09;进行操作的低级运算&#xff0c;常用于处理硬件、优化性能、加密算法或底层资源管理。以下是按位运算符的详细说明、示例和典型应用场景&#xff1a; 1.按位…

Flask与 FastAPI 对比:哪个更适合你的 Web 开发?

在开发 Web 应用时&#xff0c;Python 中有许多流行的 Web 框架可以选择&#xff0c;其中 Flask 和 FastAPI 是两款广受欢迎的框架。它们各有特色&#xff0c;适用于不同的应用场景。本文将从多个角度对比这两个框架&#xff0c;帮助你更好地选择适合的框架来构建你的 Web 应用…

Python爬虫第一战(爬取优美图库网页图片)

本文是我在学习过程中记录学习的点点滴滴,目的是为了学完之后巩固一下顺便也和大家分享一下,日后忘记了也可以方便快速的复习。 爬取网页图片 前言前言 今天学习的主要是关于如何利用Python爬取网页图片知识的理解和应用 # 1.获取网页信息,交给beautifulsoup # 2.获取页面里…

J1 ResNet-50算法实战与解析

&#x1f368; 本文為&#x1f517;365天深度學習訓練營 中的學習紀錄博客&#x1f356; 原作者&#xff1a;K同学啊 | 接輔導、項目定制 一、理论知识储备 1. 残差网络的由来 ResNet主要解决了CNN在深度加深时的退化问题&#xff08;梯度消失与梯度爆炸&#xff09;。 虽然B…

Python入门(3):语句

目录 1 基本语句 1.1 表达式语句 1.2 赋值语句 2 控制流语句 2.1 条件语句 2.2 循环语句 while循环&#xff1a; for循环&#xff1a; 2.3 流程控制语句 1. break语句&#xff1a;退出整个循环体 2. continue语句&#xff1a;只跳过本次循环&#xff0c;还会进…