【数据结构】—— 栈
- 一、栈的基本概念
- 1、栈的定义
- 2、栈的常见基本操作
- 二、栈的顺序存储
- 1、栈的顺序存储结构
- 2、顺序栈存储实现
- (1)初始化
- (2)判空
- (3)进栈
- (4)出栈
- (5)取栈顶元素
- (6)获取元素个数
- (7)销毁
- 三、栈的链式存储
- 1、栈的链式存储结构
- 2、链式栈存储实现
- (1)初始化
- (2)判空
- (3)进栈
- (4)出栈
- (5)取栈顶元素
- (6) 获取元素个数
- (7)销毁
- 四、性能分析
一、栈的基本概念
1、栈的定义
栈(stack):一种特殊的线性表,主要特性是后进先出(Last In First Out),其只允许在固定的一段进行插入和删除元素操作。进行数据插入和删除操作的一段称为栈顶,另一端称为栈底。
- 进栈:栈的插入操作叫左进栈/压栈/入栈,入数据在栈顶。
- 出栈:栈的删除操作叫做出栈。出数据也在栈顶。
- 栈顶(Top):线性表允许进行插入删除的那一端。
- 栈底(Bottom):固定的,不允许进行插入和删除的另一端。
- 空栈:不含任何元素的空表。

2、栈的常见基本操作
void STInit(ST* ps);//初始化一个空栈S。
void STDestroy(ST* ps);//栈销毁,释放其占用的存储空间//从栈顶操作
void STPush(ST* ps, STDateType x);//进栈(栈的插入操作),若栈未满,插入使x成为新栈顶
void STPop(ST* ps);//出栈(栈的删除操作),若栈非空,弹出栈顶元素
STDateType STTop(ST* ps);//返回栈顶元素
int STSize(ST* ps);//返回栈中有效元素个数(top记录有效个数)bool STEmpty(ST* ps);//判断栈是否为空(top是否指向0)
栈主要有两种存储方式来实现:一是顺序存储结构(数组),二是链式存储结构(链表)
二、栈的顺序存储
栈的顺序存储结构通常由数组和一个记录栈顶元素的下一个位置的变量(下一个入栈的数据插入的位置)组成,但是当数组空间大小不够时还需要扩容,所以还要存储栈的空间大小的变量。
可以参考我写的另一篇博客理解——顺序表

1、栈的顺序存储结构
采用顺序(数组)存储的栈称为顺序栈,它利用一组地址连续的存储单元存放自栈底到栈顶的数据元素,同时附设一个指针(top)指示当前栈顶元素的位置,一个整形(capacity)用于扩容,实现动态栈。
typedef int STDateType;//STDateType的类型根据实际情况而定
typedef struct Stack
{STDateType* a;int top;//指向栈顶下一个位置的指针->方便进栈/出栈/判空等操作int capacity;//空间大小->用于动态扩容
}ST;
2、顺序栈存储实现
(1)初始化
用一级指针接收栈的地址,将栈的存储空间(即a这个动态分配内存的数组)置为NULL,表示当前没有分配任何内存,将top和capacity置为0来表示栈的起始状态(无元素插入和无内存分配)。
void STInit(ST* ps)
{assert(ps);//断言,确保传入的指针 ps 不为空,以避免对无效指针的操作ps->a = NULL;ps->top = 0;//指向栈顶元素下一个位置ps->capacity = 0;
}
(2)判空
判断头指针的下一个位置是否指向0即可,如果指向0说明栈为空(无元素插入),否则说什么栈不为空(已有元素插入)
bool STEmpty(ST* ps)
{assert(ps);return ps->top == 0;//为真返回1,说明栈为空,否则返回0,说明栈不为空
}
(3)进栈
- 判断是否需要扩容:如果栈中有效元素(top指向的位置)和栈空间大小相同(
ps->top == ps->capacity),则需要扩容,否则无需扩容。
扩容:使用三目运算符,如果capactiy是0则置空间大小为4,否则空间大小翻倍。随后使用realloc动态内存分配出新空间给临时数组tmp,最后更新a数组和空间。 - 将x元素放入a数组,有效元素加1。
void STPush(ST* ps, STDateType x)
{assert(ps);//满了,扩容if (ps->top == ps->capacity){int newcapacity = ps->capacity == 0 ? 4 : ps->capacity * 2;STDateType* tmp = (STDateType*)realloc(ps->a, newcapacity * sizeof(STDateType));//STDateType* tmp = ps;if (tmp == NULL){perror("realloc fail");return;}ps->a = tmp;ps->capacity = newcapacity;}//top始终指向栈顶元素//动态内存分配:包含 N 个 STDateType 类型元素的内存块,可以像数组一样使用。ps->a[ps->top] = x;ps->top++;
}
注意:当ps->a为NULL,realloc和malloc作用相同。
- realloc(ptr, size):用于调整 ptr 指向的内存块的大小为 size。如果 ptr 是非空的,它会尝试在原有内存块上重新分配内存。如果 ptr 为空,realloc() 就相当于 malloc(size),即分配一个新的内存块。
- malloc(size):用于分配一个大小为 size 的内存块,并返回指向这块内存的指针。
(4)出栈
- 判断栈是否为空:如果为空无法出栈
- top指针-1
void STPop(ST* ps)
{assert(ps);assert(!STEmpty(ps));//调用前面的判空操作ps->top--;
}
(5)取栈顶元素
- 判断栈是否为空:如果为空无法取栈顶元素
- 返回栈顶元素 (top-1 即为栈顶元素的位置)
STDateType STTop(ST* ps)
{assert(ps);assert(!STEmpty(ps));return ps->a[ps->top - 1];
}
(6)获取元素个数
top的下标即为栈中有效元素的个数,直接返回即可
int STSize(ST* ps)
{assert(ps);return ps->top;
}
(7)销毁
释放栈占用的存储空间,置栈顶指针top和栈的空间大小置为0即可。
void STDestroy(ST* ps)
{assert(ps);free(ps->a);ps->a = NULL;ps->top = ps->capacity = 0;
}
三、栈的链式存储
链式存储的栈称为链栈,通常采用单链表实现。由于栈是链式存储,每次插入时需要申请一个新节点,因此我们定义一个结构体存放指向头节点的指针和记录有效元素个数的整形size。再定义一个结构体存放节点的数据和指向下一个节点的指针。
链表具体实现过程可以参考这篇博客——单链表

1、栈的链式存储结构
定义两个结构体
- StackNode :存放节点的数据
STDataType data;+ 存放当前节点的下一个节点struct StackNode* next; - ST:保存栈顶元素
StackNode* top;+ 记录有效数据个数int size;
typedef int STDataType;//STDateType的类型根据实际情况而定
typedef struct StackNode {STDataType data;//当前节点的数据struct StackNode* next;//当前节点的下一个节点的指针
}StackNode;
typedef struct Stack {StackNode* top;//栈顶int size;//有效数据的个数
}ST;
2、链式栈存储实现
(1)初始化
用一级指针接收ST结构体的地址,对ST结构体中保存的头结点悬空(NULL),有效元素个数初始化为0即可
void STInit(ST* ps)
{assert(ps);//ps!=NULLps->top = NULL;ps->size = 0;
}
(2)判空
用一级指针接收ST结构体的地址,判断ST结构体中保存的头节点是否为NULL即可
bool StackEmpty(ST* ps)
{assert(ps);//ps!=NULLreturn ps->top == NULL;
}
(3)进栈
判断栈是否为空,如果为空直接将数据给头节点即可,否则需要修改指针连接并更新头节点
void StackPush(ST* ps, STDataType x)
{assert(ps);//申请一个新节点StackNode* newnode = (StackNode*)malloc(sizeof(StackNode));if (newnode == NULL){perror("malloc fail!");exit(1);}newnode->data = x;if (StackEmpty(ps)){//如果栈为空,将新节点newnode赋给栈顶ps->top = newnode;ps->top->next = NULL;}else{//如果栈不为空,更新栈顶指针并修改指针连接newnode->next = ps->top;ps->top = newnode;}//有效数据+1ps->size++;
}
(4)出栈
- 判空:出栈需要栈中有元素,因此第一步断言判空
assert(!StackEmpty(ps)); - 处理头节点的元素:如果栈中只有一个头节点,那么直接销毁栈顶
free(ps->top); ps->top = NULL;否则修改头节点指针指向,定义del存放要被删的头节点,更新头节点为下一个节点ps->top = ps->top->next,释放del指向的空间(要被删除节点的空间)并置空free(del); del = NULL; - 元素个数减1
ps->size--;
void StackPop(ST* ps)
{assert(ps);//ps!=NULLassert(!StackEmpty(ps));//栈不能为空if (ps->size == 1 || ps->top->next == NULL){//如果栈中只有一个数据,直接销毁栈顶即可free(ps->top);ps->top = NULL;}else{//如果栈的数据个数大于1,先改变栈顶指针指向,再销毁栈顶空间即可StackNode* del = ps->top;ps->top = ps->top->next;free(del);//释放空间del = NULL;//指向空}//每次出栈有效数据减一ps->size--;
}
(5)取栈顶元素
断言非空后返回top指向的元素数据
STDataType StackTop(ST* ps)
{assert(ps);//ps!=NULLassert(!StackEmpty(ps));//栈不能为空return ps->top->data;
}
(6) 获取元素个数
直接返回ST结构体中的size即可
int STSize(ST* ps)
{assert(ps);//ps!=NULLreturn ps->size;
}
(7)销毁
链式栈的销毁和顺序栈的销毁不一样,链式栈的销毁需要遍历节点一一删除(和单链表的删除操作一样)
void STDestory(ST* ps)
{assert(ps);//ps!=NULLassert(!StackEmpty(ps));//栈不能为空while (!StackEmpty(ps)){StackPop(ps);}
}
四、性能分析
时间复杂度:顺序栈和链式栈在插入、删除和访问栈顶元素等操作上的时间复杂度都是O(1)。这意味着它们在这些基本操作上的性能是相似的。
空间复杂度
顺序栈
顺序栈的空间复杂度是O(n),其中n是栈中元素的数量。这是因为顺序栈使用数组来存储元素,而数组的大小确定。尽管数组中可能有一部分空间未被使用(特别是当栈未满时),但整个数组的空间都被分配给了顺序栈。如果栈的容量不足,还需要进行扩容操作,这可能会进一步增加空间复杂度。链式栈
链式栈的空间复杂度也是O(n),但这里的n指的是栈中实际元素的数量,而不是预先分配的空间大小。链式栈通过动态地创建和删除节点来管理内存,因此它不会浪费未使用的空间。然而,每个节点除了存储数据外,还需要额外的空间来存储指向下一个节点的指针,这增加了每个节点的空间开销。但从整体上看,链式栈的空间复杂度仍然是线性的,因为它只与栈中元素的数量有关。
总结:如果栈的使用过程中元素变化不可预料,有时很小,有时非常大,那么最好是用链栈,反之,如果它的变化在可控范围内,建议使用顺序栈会更好一些。
相关文章:
【数据结构】—— 栈
一、栈的基本概念1、栈的定义2、栈的常见基本操作 二、栈的顺序存储1、栈的顺序存储结构2、顺序栈存储实现(1)初始化(2)判空(3)进栈(4)出栈(5)取栈顶元素&…...
Kafka服务端日志详解
文章目录 服务端日志Topic消息存储方式主体介绍log文件追加记录消息index和timeindex索引文件 日志文件清理Kafka的文件高效读写机制Kafka的文件结构顺序写磁盘零拷贝 合理配置刷盘频率客户端消费进度管理 服务端日志 Kafka的日志信息是通过conf/server.properties文件中的log…...
C++ 数据语义学——进程内存空间布局
进程内存空间布局 1. 栈(堆栈/栈区)2. 堆(堆区)3. BSS段4. 数据段5. 代码段进程内存空间布局示意图可执行文件的内存布局示例代码 当把一个可执行文件加载到内存后,就变成了一个进程。这个虚拟空间(内存&am…...
【数据结构】六、图:2.邻接矩阵、邻接表(有向图、无向图、带权图)
二、存储结构 文章目录 二、存储结构❗1.邻接矩阵1.1无向图❗邻接矩阵-无向图代码-C 1.2有向图❗邻接矩阵-有向图代码-C 1.3带权图1.4性能分析1.5相乘 ❗2.邻接表2.1无向图2.2有向图❗邻接表-C 邻接矩阵VS邻接表邻接矩阵邻接表 ❗1.邻接矩阵 图的邻接矩阵(Adjacency Matrix) 存…...
财务会计与管理会计(三)
文章目录 销售回款提成表MATCH函数的模糊查询在提成类业务中的应用 营业收入分类数据分析OFFSET函数在制作图表数据中的应用 自动生成销售记录对账单VLOOKUP函数的应用 销售回款提成表 MATCH函数的模糊查询在提成类业务中的应用 G3INDEX(I$1:M$1,MATCH(E3,H3:M3,1)) G3INDEX(…...
【数据结构和算法】(基础篇三)——栈和队列
栈和队列 栈(Stack)和队列(Queue)是两种非常基本的数据结构,它们主要用于存储和检索元素。尽管它们都用于管理一组数据项,但它们的访问规则和数组都是不同的。 栈 栈是一种后进先出(Last In,…...
Linux截图工具gsnap移植arm平台过程记录
Linux截图工具gsnap移植arm平台过程记录 最近工作中一款新产品开发接近尾声,需要写文档截图产品图形,找了一款开源的Linux截屏工具gsnap,将其移植到ARM产品中,这里记录一下移植过程。 gsnap 这个工具源代码就是一个C语言源文件&a…...
密码学知识点02
#来自ウルトラマンレオ(雷欧) 1 常见加密方式 2 对称加密 采用单钥密码系统的加密方法,同一个密钥可以同时用作信息的加密和解密,这种加密方法称为对称加密,也称为单密钥加密。 常见加密算法: DES : Data…...
实现Pytest测试用例按顺序循环执行多次
要实现测试用例按顺序循环执行多次,可以使用 pytest 的自定义装饰器或插件。这里有两种方法可以实现这个需求: 方法一:使用 pytest-repeat 插件 pytest-repeat 插件允许你重复执行测试用例。你可以使用 --count 参数来指定每个测试用例的执…...
SVN工作原理和使用示例
SVN(Subversion)是另一种版本控制系统,用于管理项目文件及其变更历史。与Git不同,SVN是集中式版本控制系统,这意味着所有版本控制操作都集中在一个中央服务器上。以下是SVN的工作原理和基本使用示例。 目录 SVN 工作…...
云服务器部署Java+Vue前后端分离项目
1、申请一个云服务器 选择云服务器:阿里云、腾讯云、百度云、京东云、华为云等等,我使用的是阿里云服务器。 2、远程链接服务器 使用FinalShell工具或者其他远程工具,使用SSH链接,主机地址要填写阿里云服务的公网ip,如…...
C++的7种设计模式原则
一、设计模式前言 设计模式(Design Patterns)的“模式”指的是一种在软件设计中经过验证的、解决特定问题的方案。它们不是具体的代码,而是解决常见设计问题的抽象方案或模板。设计模式提供了一种标准的方式来组织代码,以提高代码…...
24.8.5数据结构|栈
栈-弹夹 1、定义: 栈就是特殊的线性表,与之前的线性表的区别就是增加了约束,只允许在一端插入和删除,就这麽简单。 2、基本操作 栈的插入操作叫:入栈{进栈、压栈};栈的删除:出栈{退栈&#x…...
LeetCode算法题训练
力扣刷题训练 开始记录力扣的刷题之路 刷题思路来自灵茶山艾府 入门题单: 「新」动计划 编程入门编程基础 0 到 1 训练方法 A 滑动窗口(定长/不定长/多指针)二分算法(二分答案/最小化最大值/最大化最小值/第K小)…...
Python | Leetcode Python题解之第326题3的幂
题目: 题解: class Solution:def isPowerOfThree(self, n: int) -> bool:return n > 0 and 1162261467 % n 0...
手机CPU性能天梯图(2024年8月),含安兔兔/GB6/3DMark跑分
原文地址(高清无水印原图/持续更新/含榜单出处链接): 2024年8月手机处理器天梯图 2024年8月1日更新日志:由于近期并未有新处理器发布,故只做常规更新;移除鲁大师天梯图;补充其它天梯图数量。 -…...
通过实际的例子和代码演示,可以更好地理解 `optional` 的使用方式和应用场景
当然,让我们通过一些实际的例子来演示 std::optional 的使用方式和应用场景。 场景 1:函数返回值 假设我们有一个函数,它尝试从字符串中解析一个整数,但如果字符串不是一个有效的整数,我们希望返回一个错误状态。 #…...
Java 电商秒杀系统优化实战:实现进阶示例详解与 RabbitMQ 配置
上一篇博客介绍了使用消息队列、异步处理等技术构建 Java 电商秒杀系统的基本思路,本文将进一步优化代码实现,并提供更详细的代码示例和 RabbitMQ 配置,助您构建更健壮、高效的秒杀系统。 一、 代码优化 1. 接口限流 在 SeckillController…...
路径规划 | 基于狼群算法的无人机路径规划(Matlab)
目录 效果一览基本介绍程序设计参考文献 效果一览 基本介绍 路径规划 | 基于狼群算法的无人机路径规划(Matlab) 狼是一种群居性动物,社会分工明确,通过承担各自的责任与团结协作,共同促进整个狼群的生存与发展。狼群算…...
13-python函数返回值和装包的后续提取数据方法——解包
1.1 参数解包 不定长参数简单来讲就是装包,把多个参数装到一个元组或者装到字典中,就叫做装包 Ctrld可以快速向下复制 传递实参时,也可以在序列类型的参数前添加星号,这样他会自动将序列中的元素依次作为参数传递 注意&#x…...
零基础部署Phi-4-mini推理模型:5分钟搞定数学解题AI助手
零基础部署Phi-4-mini推理模型:5分钟搞定数学解题AI助手 1. 为什么选择Phi-4-mini-reasoning? 数学解题和逻辑推理一直是AI领域的挑战性任务。传统的大型语言模型虽然功能强大,但部署成本高、响应速度慢。Phi-4-mini-reasoning作为微软推出…...
408计算机考研-计算机操作系统笔记-王道
计算机操作系统笔记-王道1.1.11.1.2操作系统的概念与功能操作系统的概念(定义)操作系统的功能和目标--向上提供方便易用的服务总结1.1.3 操作系统的特性并发与共享虚拟异步总结1.2_操作系统的发展和分类手工阶段批处理阶段--单道批处理系统多道批处理系统…...
保姆级指南:Mac上如何一键部署GLM-4.6V-Flash-WEB,实现图片智能问答
保姆级指南:Mac上如何一键部署GLM-4.6V-Flash-WEB,实现图片智能问答 1. 为什么选择GLM-4.6V-Flash-WEB? 在当今AI技术快速发展的时代,能够"看懂"图片并回答问题的多模态模型变得越来越重要。GLM-4.6V-Flash-WEB是智谱…...
实时手机检测-通用部署案例:中小企业监控场景中手机识别落地解析
实时手机检测-通用部署案例:中小企业监控场景中手机识别落地解析 1. 项目背景与价值 在现代企业管理中,手机使用管理一直是令人头疼的问题。特别是在生产车间、会议室、考场等场所,员工或学生违规使用手机不仅影响工作效率,还可…...
**为生命按下“刷新键”:当细胞科技成为健康管理的新日常**
清晨六点半,张教授在太湖边完成了他的五公里慢跑。这位年近六十的物理学博导,面色红润,步伐稳健,让许多年轻同事都自叹不如。朋友们常打趣问他保养秘诀,他总是笑笑说:“不过是尊重科学,提前管理…...
ZGC实战:如何在大内存场景下实现毫秒级GC停顿(附调优参数详解)
ZGC深度调优:TB级堆内存下的毫秒级GC实战指南 引言:大内存时代的GC挑战 在当今云计算与大数据时代,Java应用堆内存规模正经历指数级增长。从早期的GB级到如今的TB级,传统垃圾回收器如G1、CMS已无法满足低延迟需求。某头部电商平台…...
OpenClaw文件管理机器人:千问3.5-9B智能归类200+技术文档
OpenClaw文件管理机器人:千问3.5-9B智能归类200技术文档 1. 为什么需要文件管理机器人 我的下载文件夹已经变成了一个数字黑洞——里面堆积着超过200份未分类的技术文档,包括PDF白皮书、Markdown笔记、代码片段和会议录音。每次寻找特定文件都需要在混…...
嵌入式飞控信号滤波:SMA/EMA/互补滤波与卡尔曼简化实现
1. NexgenFilter 库概述:面向嵌入式飞行控制的轻量级信号处理工具集NexgenFilter 是专为 Nexgen Magpie 无人机飞控系统设计的一套高性能、低开销数字滤波与噪声生成库。它并非通用 DSP 库,而是深度嵌入在实时性严苛、资源受限的 MCU(如 STM3…...
RMBG-2.0快速上手指南:上传即处理,3步完成透明物体精细抠图
RMBG-2.0快速上手指南:上传即处理,3步完成透明物体精细抠图 1. 为什么你需要RMBG-2.0——不只是“能用”,而是“好用” 你有没有遇到过这样的情况:一张玻璃杯的照片,边缘泛着光晕,背景和杯身几乎融为一体…...
Codex CLI实战:5分钟搞定React Hooks重构与数据库迁移(附避坑指南)
Codex CLI实战:5分钟搞定React Hooks重构与数据库迁移(附避坑指南) 在快节奏的现代开发中,效率工具的价值愈发凸显。最近半年,身边不少团队开始将Codex CLI作为日常开发的"瑞士军刀"——特别是处理那些重复性…...
