【数据结构和算法】--- 栈
目录
- 栈的概念及结构
- 栈的实现
- 初始化栈
- 入栈
- 出栈
- 其他一些栈函数
- 小结
- 栈相关的题目
栈的概念及结构
栈是一种特殊的线性表。相比于链表和顺序表,栈只允许在固定的一端进行插入和删除元素操作。进行数据插入和删除操作的一端称为栈顶,另一端称为栈底。栈中的数据元素遵守后进先出LIFO(Last In First Out)的原则。
- 压栈:栈的插入操作叫做进栈/压栈/入栈,入数据在栈顶。
- 出栈:栈的删除操作叫做出栈。出数据也在栈顶。
联想一下,其实栈就相当于手枪的弹夹,将子弹压入弹夹的操作就相当于压栈,打出子弹的操作就相当于出栈,每次先打出的子弹都是我们最后压入弹夹的子弹,最后一颗子弹就是我们最先压入的那一颗了,这就是后进先出。栈也如此,结构大致如下:
基于这样的结构,那么如果我们想要拿到栈的某个元素,就必须要先把此元素以上的元素依次出栈,然后才能取出。
栈的实现
有两种方式可以实现栈结构-数组栈,链式栈:
- 链式栈
如果用单链表实现:若栈底就指向头节点,栈顶就指向尾节点。这样设计入栈很方便,相当于头插,时间复杂度为O(1)
;但出栈操作就必须要先遍历链表找到栈顶的前一个,然后再出栈,并修改栈顶,相当于尾删,时间复杂度达到O(N)
。于是乎我们一般将栈顶指向头节点,栈底指向尾节点,这样入栈出栈就都是O(1)
了,即头插/头删。
如果用双向链表实现:栈顶为链表的头和尾都可以,入栈和出栈时间复杂度都为O(1)
,但双向链表结构较为复杂,一般不选用此结构
- 数组栈
数组栈的入栈和出栈的实现较为简单,且时间复杂度为O(1)
相较于链式栈,数组栈访问数据时cpu缓存命中率比较高,但链式栈相较于数组栈也会节省一定的空间。下面栈的实现主要用的是数组栈。
通常我们标识栈顶位置的下一个位置为top
(即下标为size
的位置)。与标识栈顶位置为top
相比较,这样可以减少栈为空,栈容量判断等函数的难度,且若标识栈顶位置为top
,当栈里面没有元素时top
的指向也较为尴尬。
我们可以如下定义栈结构:
typedef int STDataType;
//数组栈
typedef struct stack
{STDataType* a;int top;//标识栈顶下一个元素下标(同为栈元素个数)int capacity;
}ST;
初始化栈
通过上面对栈的介绍进行初始化。
//初始化
void StackInit(ST* pst)
{assert(pst);pst->top = 0;pst->capacity = 0;pst->a = NULL;
}
入栈
入栈操作就是向数组内增加一个数,首先要判断栈(数组)容量pst->capacity
是否需要增容,然后向top
位置(即pst->a[top]
)增加一个数,最后重新变换top
指向(即pst->top++
),具体如下:
//入栈
void StackPush(ST* pst, STDataType x)
{assert(pst);//判断增容if (pst->top == pst->capacity){int newcapacity = pst->capacity == 0 ? 4 : pst->capacity * 2;STDataType* newnode = (STDataType*)realloc(pst->a, sizeof(ST) * newcapacity);if (newnode == NULL){perror("check_ST_capacity()::malloc");return;}pst->a = newnode;pst->capacity = newcapacity;}//目标数x入栈pst->a[pst->top] = x;//变换top指向pst->top++;
}
出栈
出栈操作就相对简单了,直接改变top
指向就可以了(即pst->top--
)。如果栈里面已经没有元素了,那执行此操作top
指向就会错误,于是乎我们需要断言一下来确保栈里面有元素可以删除(即assert(ps->top != 0);
)。
//出栈
void StackPop(ST* pst)
{assert(pst);assert(pst->top != 0);pst->top--;
}
其他一些栈函数
- 获得栈顶元素:
pst->top
指向的是栈顶的下一个元素的下标,那么只需要让他--
即可(即pst->a[pst->top-1]
),在使用前确保栈中有元素,不然程序会崩溃(越界访问)。
// 获取栈顶元素
STDataType StackTop(ST* pst)
{assert(pst);assert(pst->top != 0);return pst->a[pst->top - 1];
}
- 获得栈有效元素个数:
pst->top
指向的既是指向栈顶下一个元素的下标也是整个栈里面有效数据的个数,所以此函数返回pet->top
即可。
// 获取栈中有效元素个数
int StackSize(ST* pst)
{assert(pst);return pst->top;
}
- 检查栈是否为空:
同理只要栈里面有效元素个数为0
,那么栈就是空栈,如下:
// 检测栈是否为空,如果为空返回非零结果,如果不为空返回0
bool StackEmpty(ST* pst)
{assert(pst);return pst->top == 0;
}
- 栈的销毁:
栈的销毁本质上是释放先前realloc()
开辟的数组,再将容量和栈顶置0
即可。
// 销毁栈
void StackDestroy(ST* pst)
{assert(pst);assert(pst->capacity != 0);free(pst->a);pst->a = NULL;pst->top = pst->capacity = 0;
}
小结
-
栈是一种后进先出的结构,这一点恰与我们后面要讲的队列相反;
-
顺序表和链表都可以用来实现栈,不过一般都使用顺序表,因为栈想当于是阉割版的顺序表,只用到了顺序表的尾插和尾删操作,顺序表的尾插和尾删不需要搬移元素,因此效率非常高
O(1)
,故一般都是使用顺序表实现; -
栈结构中的
top
一般为要插入位置的下标(即栈顶元素下一个位置),这是为了方便区分栈为空栈的情况,且后续函数更好实现; -
栈只能在栈顶进行输入的插入和删除操作,不支持随机访问;
栈相关的题目
- 关于入栈和出栈顺序,如下:
若进栈序列为 1,2,3,4 ,进栈过程中可以出栈,则下列不可能的一个出栈序列是()
A 1,4,3,2
B 2,3,4,1
C 3,1,4,2
D 3,4,2,1
不难看出是c
选项错了,因为如果第一个出栈的是3
,那么在3
之前压栈的1
和2
就都还没有出栈,所以接下来出栈的只能有两种情况:
- 1.
4
接着入栈然后出栈,即为D
选项; - 2.直接出先前压栈的
2
。
对于C
选项,此时的1
还在栈底,在它上面还有2
,所以不能直接出1
。
- LeetCode OJ题: 有效的括号
题目描述:给定一个只包括
'('
,')'
,'{'
,'}'
,'['
,']'
的字符串s
,判断字符串是否有效。
有效字符串需满足:
- 左括号必须用相同类型的右括号闭合。
- 左括号必须以正确的顺序闭合。
- 每个右括号都有一个对应的相同类型的左括号。
这题主要考察我们对栈特性的应用,即后进先出,那么我们便可这样设计:循环遍历字符串s
中的每个字符,满足以下条件的对栈进行入/出栈操作:
- 遇到左括号,直接入栈;
- 遇到右括号,取栈顶元素进行匹配,若不匹配直接返回
false
,若匹配就将此括号出栈,并继续循环。
另外我们还要对如下两种情况做出判断:
- 当遍历到右括号时,此时栈中是否还有元素?(
QueueEmpty()
?)为空直接返回false
; - 当字符串
s
遍历结束时,栈中是否还有剩余元素?(QueueEmpty()
?)不为空直接返回false
,为空返回true
。
其中一些栈的接口函数就不展示了,上面内容都有,代码实现如下:
bool isValid(char* s)
{ST st;//创建栈StackInit(&st);//初始化栈//遍历字符串swhile(*s){if(*s == '(' || *s == '[' || *s == '{'){StackPush(&st,*s);}else{//栈为空判断,为空返回false,如上讲解1处if(StackEmpty(&st)){StackDestroy(&st);return false;}char ch = StackTop(&st);//左右括号匹配判断,匹配错误返回falseif((*s == ')' && ch != '(') || (*s == ']' && ch != '[') ||(*s == '}' && ch != '{')){StackDestroy(&st);return false;}StackPop(&st);}s++;}//栈为空判断,不为空返回false,与上面判断处区分,如上讲解2处if(!StackEmpty(&st)){StackDestroy(&st);return false;}return true;
}
相关文章:

【数据结构和算法】--- 栈
目录 栈的概念及结构栈的实现初始化栈入栈出栈其他一些栈函数 小结栈相关的题目 栈的概念及结构 栈是一种特殊的线性表。相比于链表和顺序表,栈只允许在固定的一端进行插入和删除元素操作。进行数据插入和删除操作的一端称为栈顶,另一端称为栈底。栈中的…...
CentOS7.0 下rpm安装MySQL5.5.60
下载 下载路径: MySQL :: Download MySQL Community Server -->looking for the latest GA version-->5.5.60 此压缩包中有多个rpm包 有四个不是必须的,只需安装这三个 MySQL-server-5.5.60-1.el6.x86_64 MySQL-devel-5.5.60-1.el6.x86_64 MySQL-client-5.5.60-1.el6.x8…...

智慧能源:数字孪生压缩空气储能管控平台
压缩空气储能在解决可再生能源不稳定性和提供可靠能源供应方面具有重要的优势。压缩空气储能,是指在电网负荷低谷期将电能用于压缩空气,在电网负荷高峰期释放压缩空气推动汽轮机发电的储能方式。通过提高能量转换效率、增加储能密度、快速启动和调节能力…...

【链表OJ—反转链表】
提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 目录 前言 1、反转链表题目: 2、方法讲解: 解法一: 解法二: 总结 前言 世上有两种耀眼的光芒,一种是正在升起的太…...

TCP一对一聊天
客户端 import java.awt.BorderLayout; import java.awt.Color; import java.awt.Dimension; import java.awt.Font; import java.awt.event.ActionEvent; import java.awt.event.ActionListener; import java.io.BufferedReader; import java.io.IOException; import java.io…...

基于Java的招聘系统的设计与实现
末尾获取源码 开发语言:Java Java开发工具:JDK1.8 后端框架:SSM 前端:Vue 数据库:MySQL5.7和Navicat管理工具结合 服务器:Tomcat8.5 开发软件:IDEA / Eclipse 是否Maven项目:是 目录…...

spring boot整合mybatis进行部门管理管理的增删改查
部门列表查询: 功能实现: 需求:查询数据库表中的所有部门数据,展示在页面上。 准备工作: 准备数据库表dept(部门表),实体类Dept。在项目中引入mybatis的起步依赖,mysql的…...

微软 Power Platform 零基础 Power Pages 网页搭建高阶实际案例实践(四)
微软 Power Platform 零基础 Power Pages 网页搭建教程之高阶案例实践学习(四) Power Pages 实际案例学习进阶 微软 Power Platform 零基础 Power Pages 网页搭建教程之高阶案例实践学习(四)1、新增视图,添加List页面2…...

如何在任何STM32上面安装micro_ros
就我知道的:micro-ros只能在特定的昂贵的开发板上面运行,但是偶然发现了这个文章,似乎提供了一个全新的方式来在ros2和单片机之间通讯,如果能够这样肯定也能够提高效率,但即使不行,使用串口库也应该比较简单…...

肖sir__ 项目讲解__项目数据
项目时间: 情况一:项目时间开始到上线的时间,这个时间一般比较长(一年,二年,三年) 情况二:项目的版本的时间或则是周期(1个月,2个月,3个月&…...

微服务实战系列之J2Cache
前言 经过近几天陆续发布Cache系列博文,博主已对业界主流的缓存工具进行了基本介绍,当然也提到了一些基本技巧。相信各位盆友看见这么多Cache工具后,在选型上一定存在某些偏爱: A同学说:不管业务千变万化,…...

12.ROS导航模块:gmapping、AMCL、map_server、move_base案例
目录 1 导航概述 2 导航简介 2.1 导航模块简介 1.全局地图 2.自身定位 3.路径规划 4.运动控制 5.环境感知 2.2 导航坐标系odom、map 1.简介 2.特点 3.坐标系变换 2.3 导航条件说明 1.硬件 2.软件 3 导航实现 3.1 创建本篇博客的功能包 3.2 建图--gmapping 3.…...

C++中string类的使用
一.string类 1.1为什么学习string类? C 语言中,字符串是以 \0 结尾的一些字符的集合,为了操作方便, C 标准库中提供了一些 str 系列的库函数,但是这些库函数与字符串是分离开的,不太符合OOP 的思想&#x…...
LeeCode每日刷题12.8
搜索插入位置 给定一个排序数组和一个目标值,在数组中找到目标值,并返回其索引。如果目标值不存在于数组中,返回它将会被按顺序插入的位置。 请必须使用时间复杂度为 O(log n) 的算法。 示例 1: 输入: nums [1,3,5,6], target 5 输出: …...

硕士毕业论文格式修改要点_word
目录 0、最开始要做的事情1、更改样式(先善器)2、多级标题(解决自动更新问题必要的基础设置)2、插入图片(1)设置一个图片样式——“无间隔”(2)插入题注(3)修…...

远红外温和护理,一贴缓解痛风不适
在冬天,很多人都会因为痛风等原因引起的关节炎症而感到不适,因为关节疼痛、肢体麻木等问题会对生活质量造成很大的影响。市场上缓解关节酸痛的护理品很多,常见的应该还是关节贴,我现在用的就是何浩明关节痛风贴。 相比于同类产品&…...

优化 SQL 日志记录的方法
为什么 SQL 日志记录是必不可少的 SQL 日志记录在数据库安全和审计中起着至关重要的作用,它涉及跟踪在数据库上执行的所有 SQL 语句,从而实现审计、故障排除和取证分析。SQL 日志记录可以提供有关数据库如何访问和使用的宝贵见解,使其成为确…...

Java设计模式-工厂模式
目录 一、简单工厂模式 (一)需求 (二)使用传统的方法来完成 (三)传统方法的优缺点 (四)基本介绍 (五)使用简单工厂模式 二、工厂方法模式 ࿰…...

每天五分钟计算机视觉:稠密连接网络(DenseNet)
本文重点 在前面的课程中我们学习了残差网络ResNet,而DenseNet可以看成是ResNet的后续,我们看一下图就可以看出二者的主要区别了。 特点 DenseNet是一种卷积神经网络,它的特点是每一层都直接连接到所有后续层。这意味着,每一层都接收来自前一层的输出,并将其作为输入传递…...
mysql支持的整数类型、各类型整数能够表示的数值范围
MySQL :: MySQL 8.2 Reference Manual :: 11.1.2 Integer Types (Exact Value) - INTEGER, INT, SMALLINT, TINYINT, MEDIUMINT, BIGINT mysql支持的整数有:TINYINT、SMALLINT、MEDIUMINT、INT(INT和INTEGER是同义词)、BIGINT,各…...
在rocky linux 9.5上在线安装 docker
前面是指南,后面是日志 sudo dnf config-manager --add-repo https://download.docker.com/linux/centos/docker-ce.repo sudo dnf install docker-ce docker-ce-cli containerd.io -y docker version sudo systemctl start docker sudo systemctl status docker …...

CMake 从 GitHub 下载第三方库并使用
有时我们希望直接使用 GitHub 上的开源库,而不想手动下载、编译和安装。 可以利用 CMake 提供的 FetchContent 模块来实现自动下载、构建和链接第三方库。 FetchContent 命令官方文档✅ 示例代码 我们将以 fmt 这个流行的格式化库为例,演示如何: 使用 FetchContent 从 GitH…...

ABAP设计模式之---“简单设计原则(Simple Design)”
“Simple Design”(简单设计)是软件开发中的一个重要理念,倡导以最简单的方式实现软件功能,以确保代码清晰易懂、易维护,并在项目需求变化时能够快速适应。 其核心目标是避免复杂和过度设计,遵循“让事情保…...

RabbitMQ入门4.1.0版本(基于java、SpringBoot操作)
RabbitMQ 一、RabbitMQ概述 RabbitMQ RabbitMQ最初由LShift和CohesiveFT于2007年开发,后来由Pivotal Software Inc.(现为VMware子公司)接管。RabbitMQ 是一个开源的消息代理和队列服务器,用 Erlang 语言编写。广泛应用于各种分布…...

力扣热题100 k个一组反转链表题解
题目: 代码: func reverseKGroup(head *ListNode, k int) *ListNode {cur : headfor i : 0; i < k; i {if cur nil {return head}cur cur.Next}newHead : reverse(head, cur)head.Next reverseKGroup(cur, k)return newHead }func reverse(start, end *ListNode) *ListN…...

2025年渗透测试面试题总结-腾讯[实习]科恩实验室-安全工程师(题目+回答)
安全领域各种资源,学习文档,以及工具分享、前沿信息分享、POC、EXP分享。不定期分享各种好玩的项目及好用的工具,欢迎关注。 目录 腾讯[实习]科恩实验室-安全工程师 一、网络与协议 1. TCP三次握手 2. SYN扫描原理 3. HTTPS证书机制 二…...

[ACTF2020 新生赛]Include 1(php://filter伪协议)
题目 做法 启动靶机,点进去 点进去 查看URL,有 ?fileflag.php说明存在文件包含,原理是php://filter 协议 当它与包含函数结合时,php://filter流会被当作php文件执行。 用php://filter加编码,能让PHP把文件内容…...

【UE5 C++】通过文件对话框获取选择文件的路径
目录 效果 步骤 源码 效果 步骤 1. 在“xxx.Build.cs”中添加需要使用的模块 ,这里主要使用“DesktopPlatform”模块 2. 添加后闭UE编辑器,右键点击 .uproject 文件,选择 "Generate Visual Studio project files",重…...

rknn toolkit2搭建和推理
安装Miniconda Miniconda - Anaconda Miniconda 选择一个 新的 版本 ,不用和RKNN的python版本保持一致 使用 ./xxx.sh进行安装 下面配置一下载源 # 清华大学源(最常用) conda config --add channels https://mirrors.tuna.tsinghua.edu.cn…...

QT开发技术【ffmpeg + QAudioOutput】音乐播放器
一、 介绍 使用ffmpeg 4.2.2 在数字化浪潮席卷全球的当下,音视频内容犹如璀璨繁星,点亮了人们的生活与工作。从短视频平台上令人捧腹的搞笑视频,到在线课堂中知识渊博的专家授课,再到影视平台上扣人心弦的高清大片,音…...