数据结构——栈的基本操作

前言

介绍

🍃数据结构专区:数据结构

参考

该部分知识参考于《数据结构(C语言版 第2版)》55 ~ 59页

🌈每一个清晨,都是世界对你说的最温柔的早安:ૢ(≧▽≦)و✨


1、栈的基本概念

栈(Stack)是一种遵循后进先出(LIFO, Last In First Out)原则的有序集合。

  • 数组实现:使用数组的一个连续空间来存储栈中的元素,通常使用一个指针(或索引)来指示栈顶的位置。数组实现的栈在添加和删除元素时,如果数组已满或为空,可能需要进行扩容或缩容操作,这可能会涉及到额外的性能开销。
  • 链表实现:使用链表的头部(或尾部,取决于具体实现)作为栈顶,通过修改链表的头指针(或尾指针)来实现元素的添加和删除。链表实现的栈在添加和删除元素时,不需要进行扩容或缩容操作,因此通常具有更好的性能。

2、数组栈的实现

2.1  宏定义

#include<iostream>
using namespace std;
#define OK 1
#define ERROR 0
#define OVERFLOW -2
typedef int Status;

2.2  数组栈的结构体定义

#define MAXSIZE 100  //顺序表的存储范围
#define STACK_INCREMENT 2  //存储空间分配增量//这里假定SElemType是int类型
typedef int SElemType;typedef struct
{SElemType* base;   //栈底指针SElemType* top;    //栈顶指针int stacksize;     //栈可用的最大容量
}SqStack;

2.3  初始化数组栈

//初始化栈
Status InitStack(SqStack& S)
{S.base = new SElemType[MAXSIZE];if (!S.base)exit(OVERFLOW);S.top = S.base;  //top初始化为base,空栈S.stacksize = MAXSIZE;  //stacksize的最大容量为MAXSIZEreturn OK;
}

 2.4  销毁数组栈

//销毁栈
Status DestroyStack(SqStack& S)
{free(S.base);S.base = NULL;S.top = NULL;  //规范指针S.stacksize = 0;return OK;
}

2.5  清空数组栈

//清除栈
Status CleanStack(SqStack& S)
{S.top = S.base; //让top指针指回栈底即可return OK;
}

2.6  判空

//判空
Status StackEmpty(SqStack S)
{if (S.top == S.base) //如果栈顶和栈底指向同一个位置则证明栈为NULLreturn OK;elsereturn ERROR;
}

2.7  获取栈内元素个数

//获取栈内元素数量
int StackLength(SqStack S)
{return S.top - S.base;
}

2.8  获取栈顶元素

//获取栈顶元素
SElemType GetTop(SqStack S)
{//判断栈是否为空if (!StackEmpty(S))//不为空即可获取元素{return *(--S.top);}else{return ERROR;}
}

2.9  入栈

//入栈
Status Push(SqStack& S, SElemType e)
{//插入元素为新的栈顶元素if (S.top - S.base == S.stacksize)//满栈{S.base = (SElemType*)realloc(S.base, (S.stacksize + STACK_INCREMENT) * sizeof(SElemType));//判断是否扩容成功if (!S.base)exit(OVERFLOW);//如果扩容失败,退出程序S.top = S.base + S.stacksize;}*(S.top)++ = e;  //这里是先对S.top解引用后存入数据e,随后将top指针向后移动一位
}

2.10  出栈

//出栈
Status Pop(SqStack& S, SElemType &e)
{//删除栈顶元素,并返回其值if (!StackEmpty(S)) //栈不为空进行操作{e = *(--S.top);return OK;}elsereturn ERROR;
}

2.11  visit()函数

// 定义一个函数visit,用于打印元素
void visit(SElemType e)
{std::cout << e << " ";
}

2.12  遍历数组栈

// 定义一个函数用于遍历栈中的元素并对每个元素执行visit函数
void StackTraverse(SqStack S, void(*visit)(SElemType)) 
{SElemType* p = S.base;while (S.top > p) //p指向栈元素visit(*p++); //对该栈调用visit(),p指针上移一个存储单元printf("\n");
}

2.13  整体代码(含测试代码)

#include<iostream>
using namespace std;
#define OK 1
#define ERROR 0
#define OVERFLOW -2
typedef int Status;#define MAXSIZE 100  //顺序表的存储范围
#define STACK_INCREMENT 2  //存储空间分配增量//这里假定SElemType是int类型
typedef int SElemType;typedef struct
{SElemType* base;   //栈底指针SElemType* top;    //栈顶指针int stacksize;     //栈可用的最大容量
}SqStack;//SqStack S;  //声明该栈//初始化栈
Status InitStack(SqStack& S)
{S.base = new SElemType[MAXSIZE];if (!S.base)exit(OVERFLOW);S.top = S.base;  //top初始化为base,空栈S.stacksize = MAXSIZE;  //stacksize的最大容量为MAXSIZEreturn OK;
}//销毁栈
Status DestroyStack(SqStack& S)
{free(S.base);S.base = NULL;S.top = NULL;  //规范指针S.stacksize = 0;return OK;
}//清除栈
Status CleanStack(SqStack& S)
{S.top = S.base; //让top指针指回栈底即可return OK;
}//判空
Status StackEmpty(SqStack S)
{if (S.top == S.base) //如果栈顶和栈底指向同一个位置则证明栈为NULLreturn OK;elsereturn ERROR;
}//获取栈内元素数量
int StackLength(SqStack S)
{return S.top - S.base;
}//获取栈顶元素
SElemType GetTop(SqStack S)
{//判断栈是否为空if (!StackEmpty(S))//不为空即可获取元素{return *(--S.top);}else{return ERROR;}
}//入栈
Status Push(SqStack& S, SElemType e)
{//插入元素为新的栈顶元素if (S.top - S.base == S.stacksize)//满栈{S.base = (SElemType*)realloc(S.base, (S.stacksize + STACK_INCREMENT) * sizeof(SElemType));//判断是否扩容成功if (!S.base)exit(OVERFLOW);//如果扩容失败,退出程序S.top = S.base + S.stacksize;}*(S.top)++ = e;  //这里是先对S.top解引用后存入数据e,随后将top指针向后移动一位
}//出栈
Status Pop(SqStack& S, SElemType &e)
{//删除栈顶元素,并返回其值if (!StackEmpty(S)) //栈不为空进行操作{e = *(--S.top);return OK;}elsereturn ERROR;
}// 定义一个函数visit,用于打印元素
void visit(SElemType e)
{std::cout << e << " ";
}// 定义一个函数用于遍历栈中的元素并对每个元素执行visit函数
void StackTraverse(SqStack S, void(*visit)(SElemType)) 
{SElemType* p = S.base;while (S.top > p) //p指向栈元素visit(*p++); //对该栈调用visit(),p指针上移一个存储单元printf("\n");
}int main() {int j;SqStack s;SElemType e;InitStack(s);for (j = 1; j <= 12; j++)Push(s, j);printf("栈中元素依次为\n");StackTraverse(s, visit);Pop(s, e);printf("弹出的栈顶元素e = %d\n", e);printf("栈空否? %d (1:空 0:否)\n", StackEmpty(s));e = GetTop(s);printf("栈顶元素e = %d,栈的长度为%d\n", e, StackLength(s));CleanStack(s);printf("清空栈后,栈空否? %d (1:空 0:否)\n", StackEmpty(s));DestroyStack(s);printf("销毁栈后,s.top = %u,s.base = %u,s.stacksize = %d\n", s.top, s.base, s.stacksize);
}

3、链表栈的实现

3.1  宏定义

#include<iostream>
using namespace std;
#define OK 1
#define ERROR 0
#define OVERFLOW -2
typedef int Status;

3.2  链表栈的结构体定义

typedef int SElemType;
typedef struct StackNode
{SElemType data;struct StackNode* next;
}StackNode, *LinkStack;

3.3  初始化链表栈

//初始化
Status InitStack(LinkStack& S)
{//让栈顶指针指向NULL即可S = NULL;return OK;
}

3.4  清空栈

//清空栈
Status ClearStack(LinkStack& S)
{//创建一个临时指针,遍历该链表后依次释放各个节点StackNode* p;while (S){p = S;S = S->next;delete p;}return OK;
}

3.5  判空

//判空
Status StackEmpty(LinkStack S)
{return S == NULL;
}

3.6  销毁链表栈

//销毁
Status DestroyStack(LinkStack& S)
{ClearStack(S);S = NULL;return OK;
}

3.7  入栈

//入栈
Status Push(LinkStack& S, SElemType e)
{//在栈顶位置插入元素eStackNode* p = new StackNode;p->data = e;p->next = S;  //将新结点插入栈顶S = p;        //修改栈顶指针为preturn OK;
}

3.8  出栈

//出栈
Status Pop(LinkStack &S, SElemType& e)
{//删除栈顶元素,并返回该元素if (S == NULL)return ERROR;StackNode * p = S;e = p->data;S = S->next;delete p;return OK;
}

 3.9  获取栈顶元素

//获取栈顶元素
SElemType GetTop(LinkStack S)
{//返回S的栈顶元素,并不改变栈顶指针的位置if (S != NULL){return S->data;}
}

3.10  遍历打印

// 遍历栈并打印
void StackTraverse(LinkStack S) 
{StackNode* p = S;while (p) {printf("%d ", p->data);p = p->next;}printf("\n");
}

3.11  整体代码(含测试代码)

#include<iostream>
using namespace std;
#define OK 1
#define ERROR 0
#define OVERFLOW -2
typedef int Status;typedef int SElemType;
typedef struct StackNode
{SElemType data;struct StackNode* next;
}StackNode, *LinkStack;//初始化
Status InitStack(LinkStack& S)
{//让栈顶指针指向NULL即可S = NULL;return OK;
}//清空栈
Status ClearStack(LinkStack& S)
{//创建一个临时指针,遍历该链表后依次释放各个节点StackNode* p;while (S){p = S;S = S->next;delete p;}return OK;
}//判空
Status StackEmpty(LinkStack S)
{return S == NULL;
}//销毁
Status DestroyStack(LinkStack& S)
{ClearStack(S);S = NULL;return OK;
}//入栈
Status Push(LinkStack& S, SElemType e)
{//在栈顶位置插入元素eStackNode* p = new StackNode;p->data = e;p->next = S;  //将新结点插入栈顶S = p;        //修改栈顶指针为preturn OK;
}//出栈
Status Pop(LinkStack &S, SElemType& e)
{//删除栈顶元素,并返回该元素if (S == NULL)return ERROR;StackNode * p = S;e = p->data;S = S->next;delete p;return OK;
}//获取栈顶元素
SElemType GetTop(LinkStack S)
{//返回S的栈顶元素,并不改变栈顶指针的位置if (S != NULL){return S->data;}
}// 遍历栈并打印
void StackTraverse(LinkStack S) 
{StackNode* p = S;while (p) {printf("%d ", p->data);p = p->next;}printf("\n");
}int main()
{LinkStack S;InitStack(S);int e;Push(S, 1);Push(S, 2);Push(S, 3);printf("现在栈内元素为(后进先出):");StackTraverse(S);printf("栈顶元素为:%d\n", GetTop(S));Pop(S, e);printf("现在栈内元素为(后进先出):");StackTraverse(S);printf("弹出一个元素后,栈顶元素为:%d\n", GetTop(S));ClearStack(S);if (StackEmpty(S)) {printf("清空栈后,栈为空\n");}else {printf("清空栈后,栈不为空,证明有问题\n");}DestroyStack(S);return 0;
}

结语

到此我们的两种栈的实现代码就完成了,我们可以发现,在实现栈的过程中远没有当初学习顺序表那么困难,那是因为栈其实就是一种特殊结构的顺序表,并且在前面的学习顺序表过程中,我故意将ElemType写为一种结构体类型,让我们在学习顺序表时写起来就有些困难,在前面学会之后看到这里就游刃有余了!

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

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

相关文章

【SpringBoot详细教程】-08-MybatisPlus详细教程以及SpringBoot整合Mybatis-plus【持续更新】

目录 🌲 MyBatis Plus 简介 🌾入门案例 🌾 MP 简介 🌲 MP 的CRUD 🌾 新增 🌾 删除 🌾 修改在进行 🌾 根据ID查询 🌾 查询所有 🌲 分页功能 🌾 设置分页参数 🌾 设置分页拦截器 🌲 优化启动 🌾 取消mbatisPlusBanner 🌾 取消Sprin…

[linux] 磁盘清理相关

在 CentOS 7 中清理磁盘空间可以通过多种方法实现&#xff0c;以下是一些常用的步骤和命令&#xff1a; 1. 查找和删除大文件 你可以使用 find 命令查找占用大量空间的文件&#xff1a; find / -type f -size 100M 2>/dev/null这条命令会查找大于 100 MB 的文件。你可以根…

国外电商系统开发-运维系统功能清单开发

一、最终效果图 二、功能清单 功能 描述 自定义日志绘图 根据Nginx、Apache登录日志文件绘图&#xff0c;绘图数据包括&#xff1a;访问量走势&#xff0c;500错误&#xff0c;200正确百分比等 创建服务器 加入服务器 主机状态自动检查 加入主机到系统后&#xff0c;系统…

LeetCode: 1971. 寻找图中是否存在路径

寻找图中是否存在路径 原题 有一个具有 n 个顶点的 双向 图&#xff0c;其中每个顶点标记从 0 到 n - 1&#xff08;包含 0 和 n - 1&#xff09;。图中的边用一个二维整数数组 edges 表示&#xff0c;其中 edges[i] [ui, vi] 表示顶点 ui 和顶点 vi 之间的双向边。 每个顶点…

记OpenGL的Demo中增加ImGui后界面不显示的问题

百思不得其解&#xff1a; 我的ImGui界面呢&#xff1f; 经过不断调整代码&#xff0c;也可能是对这还不太熟悉吧&#xff0c;最终将drawData放在clearColor后界面出来了&#xff0c;仔细想想也能理解&#xff0c;先画了界面&#xff0c;再渲染了背景色&#xff0c;将界面盖住了…

浅谈stm32的GPIO引脚配置模式

STM32的GPIO&#xff08;通用输入输出&#xff09;引脚可以被配置为多种模式&#xff0c;以适应不同的应用场景。下面介绍一些一些常见的STM32 GPIO引脚模式&#xff1a; 模拟输入模式&#xff08;Analog Input Mode&#xff09;&#xff1a;在这种模式下&#xff0c;GPIO引脚被…

Docker Compose 部署大模型GPU集群:高效分配与管理算力资源

Docker Compose 部署大模型GPU集群&#xff1a;高效分配与管理算力资源 文章目录 Docker Compose 部署大模型GPU集群&#xff1a;高效分配与管理算力资源一 Dockerfile 编写二 Dockerfile 示例三 分配GPU资源1&#xff09;GPU分配&#xff1a;指定count2&#xff09;GPU分配&am…

虚商目前有哪些业务痛点?

虚拟运营商当前面临的业务痛点主要集中在市场竞争、运营成本、技术依赖、用户体验及政策监管等方面。 一、市场竞争激烈 1、竞争者数量增加: 随着市场准入门槛的降低&#xff0c;越来越多的企业进入虚拟运营商市场导致市场竟争日益激烈。为了争夺市场份额&#xff0c;企业不得不…

【Xcode Command Line Tools】安装指南

安装指令 xcode-select --install安装 完成安装 验证 $ xcode-select -p /Library/Developer/CommandLineTools

使用Chrome浏览器时打开网页如何禁用缓存

缓存是浏览器用于临时存储网页资源的一种机制&#xff0c;可以提高网页加载速度和减轻服务器负载。 然而&#xff0c;有时候我们需要阻止缓存中的Chrome浏览器&#xff0c;以便获取最新的网页内容。以下是一些方法可以实现这个目标&#xff1a; 1、强制刷新页面&#xff1a;在C…

【Spring】运行Spring Boot项目,请求响应流程分析以及404和500报错

1. 运行项目 import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.SpringBootApplication; SpringBootApplication public class Application { public static void main(String[] args) { SpringApplication.run(Appl…

【车载开发系列】ParaSoft单元测试环境配置(四)

【车载开发系列】ParaSoft单元测试环境配置&#xff08;四&#xff09; 【车载开发系列】ParaSoft单元测试环境配置&#xff08;四&#xff09; 【车载开发系列】ParaSoft单元测试环境配置&#xff08;四&#xff09;一. 如何设置过滤二. 如何设置静态扫描的规则三. 如何设置单…

Pycharm 本地搭建 stable-diffusion-webui

一、下载工程源码 Github官方连接 https://github.com/AUTOMATIC1111/stable-diffusion-webui 二、Pycharm打开工程 1、设置环境 文件-设置-项目-Python解析器-添加解释器-添加本地解释器 Conda环境-创造新环境-Python版本3.10 注意一定要选择Python3.10版本&#xff0c;否…

tushare库简介及数据获取方法

文章目录 tushare库简介tushare使用方法接口数据获取实例获取交易日历股票基础信息沪股通和深股通成份股上市公司基础信息获取新股上市列表数据A股日线行情A股周线行情A股月线行情股票每日指标&#xff0c;选股分析 tushare库简介 tushare库是一个用于抓取中国股票的历史和实时…

虚拟机三种网络模式详解

在电脑里开一台虚拟机&#xff0c;是再常见不过的操作了。无论是用虚拟机玩只有旧版本系统能运行的游戏&#xff0c;还是用来学习Linux、跑跑应用程序都是很好的。而这其中&#xff0c;虚拟机网络是绝对绕不过去的。本篇文章通俗易懂的介绍了常见的虚拟网络提供的三种网络链接模…

从0开始实现es6 promise类

主要由基础实现和静态类的实现两部分组成。 1 基础实现&#xff08;不含静态类&#xff09; 1.1 使用类实现完成构造函数 实现代码如下&#xff0c;构造函数传入一个回调函数&#xff0c;定义resolve和reject函数&#xff0c;将两个函数作为参数执行回调函数。 // 1. 使用类实…

Java面试:ArrayList 和 LinkedList 的区别是什么?谈谈你对ArrayList和LinkedList的理解

在 Java 的集合框架中&#xff0c;ArrayList 和 LinkedList 是两种常用的 List 实现。它们各具特点&#xff0c;适用于不同的场景。本文将深入探讨这两种数据结构的概念、用法、联系和区别&#xff0c;帮助你选择合适的集合类来满足特定需求。 一、基本介绍 1. ArrayList Ar…

C/C++进阶(一)--内存管理

更多精彩内容..... &#x1f389;❤️播主の主页✨&#x1f618; Stark、-CSDN博客 本文所在专栏&#xff1a; 学习专栏C语言_Stark、的博客-CSDN博客 其它专栏&#xff1a; 数据结构与算法_Stark、的博客-CSDN博客 ​​​​​​项目实战C系列_Stark、的博客-CSDN博客 座右铭&a…

D3.js数据可视化基础——基于Notepad++、IDEA前端开发

实验:D3.js数据可视化基础 1、实验名称 D3数据可视化基础 2、实验目的 熟悉D3数据可视化的使用方法。 3、实验原理 D3 的全称是(Data-Driven Documents),是一个被数据驱动的文档,其实就是一个 JavaScript 的函数库,使用它主要是用来做数据可视化的。本次实…

【前沿 热点 顶会】NIPS 2024中目标检测有关的论文

迈向灵活的 3D 感知&#xff1a;以对象为中心的占领完成增强长序列的 3D 目标检测 虽然 3D 对象边界框(BBox)表示在自动驾驶感知中得到了广泛的应用&#xff0c;但它们缺乏捕捉对象内部几何形状的复杂细节的能力。最近&#xff0c;占有&#xff08;occupancy&#xff09;已经成…