【数据结构和算法】--- 栈
目录
- 栈的概念及结构
- 栈的实现
- 初始化栈
- 入栈
- 出栈
- 其他一些栈函数
- 小结
- 栈相关的题目
栈的概念及结构
栈是一种特殊的线性表。相比于链表和顺序表,栈只允许在固定的一端进行插入和删除元素操作。进行数据插入和删除操作的一端称为栈顶,另一端称为栈底。栈中的数据元素遵守后进先出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,各…...
《Qt C++ 与 OpenCV:解锁视频播放程序设计的奥秘》
引言:探索视频播放程序设计之旅 在当今数字化时代,多媒体应用已渗透到我们生活的方方面面,从日常的视频娱乐到专业的视频监控、视频会议系统,视频播放程序作为多媒体应用的核心组成部分,扮演着至关重要的角色。无论是在个人电脑、移动设备还是智能电视等平台上,用户都期望…...
Opencv中的addweighted函数
一.addweighted函数作用 addweighted()是OpenCV库中用于图像处理的函数,主要功能是将两个输入图像(尺寸和类型相同)按照指定的权重进行加权叠加(图像融合),并添加一个标量值&#x…...
CocosCreator 之 JavaScript/TypeScript和Java的相互交互
引擎版本: 3.8.1 语言: JavaScript/TypeScript、C、Java 环境:Window 参考:Java原生反射机制 您好,我是鹤九日! 回顾 在上篇文章中:CocosCreator Android项目接入UnityAds 广告SDK。 我们简单讲…...
新能源汽车智慧充电桩管理方案:新能源充电桩散热问题及消防安全监管方案
随着新能源汽车的快速普及,充电桩作为核心配套设施,其安全性与可靠性备受关注。然而,在高温、高负荷运行环境下,充电桩的散热问题与消防安全隐患日益凸显,成为制约行业发展的关键瓶颈。 如何通过智慧化管理手段优化散…...
【Java_EE】Spring MVC
目录 Spring Web MVC 编辑注解 RestController RequestMapping RequestParam RequestParam RequestBody PathVariable RequestPart 参数传递 注意事项 编辑参数重命名 RequestParam 编辑编辑传递集合 RequestParam 传递JSON数据 编辑RequestBody …...
IP如何挑?2025年海外专线IP如何购买?
你花了时间和预算买了IP,结果IP质量不佳,项目效率低下不说,还可能带来莫名的网络问题,是不是太闹心了?尤其是在面对海外专线IP时,到底怎么才能买到适合自己的呢?所以,挑IP绝对是个技…...
mac 安装homebrew (nvm 及git)
mac 安装nvm 及git 万恶之源 mac 安装这些东西离不开Xcode。及homebrew 一、先说安装git步骤 通用: 方法一:使用 Homebrew 安装 Git(推荐) 步骤如下:打开终端(Terminal.app) 1.安装 Homebrew…...
Python 实现 Web 静态服务器(HTTP 协议)
目录 一、在本地启动 HTTP 服务器1. Windows 下安装 node.js1)下载安装包2)配置环境变量3)安装镜像4)node.js 的常用命令 2. 安装 http-server 服务3. 使用 http-server 开启服务1)使用 http-server2)详解 …...
WebRTC从入门到实践 - 零基础教程
WebRTC从入门到实践 - 零基础教程 目录 WebRTC简介 基础概念 工作原理 开发环境搭建 基础实践 三个实战案例 常见问题解答 1. WebRTC简介 1.1 什么是WebRTC? WebRTC(Web Real-Time Communication)是一个支持网页浏览器进行实时语音…...
关于easyexcel动态下拉选问题处理
前些日子突然碰到一个问题,说是客户的导入文件模版想支持部分导入内容的下拉选,于是我就找了easyexcel官网寻找解决方案,并没有找到合适的方案,没办法只能自己动手并分享出来,针对Java生成Excel下拉菜单时因选项过多导…...


