【数据结构和算法】--- 栈
目录
- 栈的概念及结构
- 栈的实现
- 初始化栈
- 入栈
- 出栈
- 其他一些栈函数
- 小结
- 栈相关的题目
栈的概念及结构
栈是一种特殊的线性表。相比于链表和顺序表,栈只允许在固定的一端进行插入和删除元素操作。进行数据插入和删除操作的一端称为栈顶,另一端称为栈底。栈中的数据元素遵守后进先出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,各…...
盘古信息PCB行业解决方案:以全域场景重构,激活智造新未来
一、破局:PCB行业的时代之问 在数字经济蓬勃发展的浪潮中,PCB(印制电路板)作为 “电子产品之母”,其重要性愈发凸显。随着 5G、人工智能等新兴技术的加速渗透,PCB行业面临着前所未有的挑战与机遇。产品迭代…...
什么是库存周转?如何用进销存系统提高库存周转率?
你可能听说过这样一句话: “利润不是赚出来的,是管出来的。” 尤其是在制造业、批发零售、电商这类“货堆成山”的行业,很多企业看着销售不错,账上却没钱、利润也不见了,一翻库存才发现: 一堆卖不动的旧货…...
【项目实战】通过多模态+LangGraph实现PPT生成助手
PPT自动生成系统 基于LangGraph的PPT自动生成系统,可以将Markdown文档自动转换为PPT演示文稿。 功能特点 Markdown解析:自动解析Markdown文档结构PPT模板分析:分析PPT模板的布局和风格智能布局决策:匹配内容与合适的PPT布局自动…...
(二)原型模式
原型的功能是将一个已经存在的对象作为源目标,其余对象都是通过这个源目标创建。发挥复制的作用就是原型模式的核心思想。 一、源型模式的定义 原型模式是指第二次创建对象可以通过复制已经存在的原型对象来实现,忽略对象创建过程中的其它细节。 📌 核心特点: 避免重复初…...
微软PowerBI考试 PL300-在 Power BI 中清理、转换和加载数据
微软PowerBI考试 PL300-在 Power BI 中清理、转换和加载数据 Power Query 具有大量专门帮助您清理和准备数据以供分析的功能。 您将了解如何简化复杂模型、更改数据类型、重命名对象和透视数据。 您还将了解如何分析列,以便知晓哪些列包含有价值的数据,…...
听写流程自动化实践,轻量级教育辅助
随着智能教育工具的发展,越来越多的传统学习方式正在被数字化、自动化所优化。听写作为语文、英语等学科中重要的基础训练形式,也迎来了更高效的解决方案。 这是一款轻量但功能强大的听写辅助工具。它是基于本地词库与可选在线语音引擎构建,…...
算法岗面试经验分享-大模型篇
文章目录 A 基础语言模型A.1 TransformerA.2 Bert B 大语言模型结构B.1 GPTB.2 LLamaB.3 ChatGLMB.4 Qwen C 大语言模型微调C.1 Fine-tuningC.2 Adapter-tuningC.3 Prefix-tuningC.4 P-tuningC.5 LoRA A 基础语言模型 A.1 Transformer (1)资源 论文&a…...
代码随想录刷题day30
1、零钱兑换II 给你一个整数数组 coins 表示不同面额的硬币,另给一个整数 amount 表示总金额。 请你计算并返回可以凑成总金额的硬币组合数。如果任何硬币组合都无法凑出总金额,返回 0 。 假设每一种面额的硬币有无限个。 题目数据保证结果符合 32 位带…...
LLMs 系列实操科普(1)
写在前面: 本期内容我们继续 Andrej Karpathy 的《How I use LLMs》讲座内容,原视频时长 ~130 分钟,以实操演示主流的一些 LLMs 的使用,由于涉及到实操,实际上并不适合以文字整理,但还是决定尽量整理一份笔…...
计算机基础知识解析:从应用到架构的全面拆解
目录 前言 1、 计算机的应用领域:无处不在的数字助手 2、 计算机的进化史:从算盘到量子计算 3、计算机的分类:不止 “台式机和笔记本” 4、计算机的组件:硬件与软件的协同 4.1 硬件:五大核心部件 4.2 软件&#…...


