《数据结构》--栈【概念应用、图文并茂】

 本节讲完栈下次再讲一下队列,最后补充一个串,我们的线性结构基本就完事了。下图中黄色框框圈中的是我们今日份内容(分为两篇博客):

知识体系图

栈(Stack-LIFO)结构

栈的基础概念

栈(Stack)是一个后进先出(Last-In-First-Out)的一个特殊数据结构,只有一端可以操作,三面环墙。按照存储方式可以分为顺序存储的栈和链式存储的栈(即顺序栈和链栈),另外还有一些特殊的栈需要我们了解:单调栈、共享栈。

不管什么栈,它的基本API接口(基本操作)都是:入栈(Push),出栈(Pop),查看栈顶元素(Top/Peek),判断栈是否为空(isEmpty)。

栈顶(top):处于可以插入或删除元素的一端,栈顶指针指向非空元素的下一个位置

栈底(base):处于不能插入或删除元素的另一端,栈底指针固定指向。

空栈(empty):栈内不包含任意元素,栈顶指针与栈底指针指向同一位置。

顺序栈

采用顺序结构存储的栈称为顺序栈,它的底层是数组,使用的是一段地址连续的空间。另外找一个尾指针充当栈顶指针,指向栈顶元素的位置的下一个地址。判断栈为空的条件就是栈顶指针指向栈底。

1.栈的顺序存储

站的顺序存储结构可以定义为:一个栈底指针,一个栈顶指针

#define INIT_MAX_SIZE 20 //初始化栈时给的最大容量
typedef int DataType;//DataType的类型不确定,根据习惯初始假定为int
typedef struct{DataType* base;DataType* top;
}SqStack;

 2.栈的基本操作

初始化

初始化时,使用预先设置好的初值来开辟大小,并且初始一定是空栈,所以让栈顶指针指向栈底即可。

void initStack(SqStack* s)
{s->base = (DataType*)malloc(sizeof(DataType)*INIT_MAX_SIZE);s->top = s->base;
}

判空

如果栈顶指针与栈底指针指向同一块空间,那么栈就是空的,直接返回。这里传不传指针都可以,不需要进行修改的操作,只进行判断,所以利用值传递也能实现目的

#include <stdbool.h>
bool isEmpty(SqStack s){return s.base == s.top;
}
bool isEmpty(SqStack* s){return s->base == s->top;
}

进栈 

进栈之前,我们先判断栈是否为满,如果满栈,我们不能随便加入,需要扩容,我们选择二倍扩容的方式,如果扩容失败,结束程序,一旦我们扩容成功,我们原来的栈顶指针也会失效,使用栈底指针加上原来的满栈数,跨越总步长回到栈顶的相对位置。然后将栈顶设置为元素e,并自动跨越一个步长(++)。

/*插入元素e为新的栈顶元素*/
void Push(SqStack *s, DataType e){//满栈if(s->top == s->base + INIT_MAX_SIZE){int size = INIT_MAX_SIZE*2;s->base = (DataType*)realloc(s->base,sizeof(DataType)*size);assert(s->base);s->top = s->base + size/2;}*(s->top) = e;    //将新插入元素赋值给栈顶空间s->top++; //更改栈顶指针//可以合并为*(s->top++)=e;
}

出栈

出栈就是要后面的元素无法访问,那么只需要将栈顶指针缩减一个步长(--)即可,但在此之前,我们需要判断是不是空栈,如果是空栈,我们就没必要去进行出栈的操作了。

/*若栈不空,则删除S的栈顶元素*/
void Pop(SqStack *s){if(s->top == s->base){return;}S->top--;   //栈顶指针减1
}

 获取栈顶元素

获取栈顶元素,我们需要获取的栈顶指针上一个位置内存储的内容,所以我们进行完判空操作后,我们需要对s->top-1进行解引用(即取*)的操作。

/*读栈顶元素*/
DataType Top(SqStack s){if(s->top == s->base){   //栈空return -1;}return *(s->top-1);   //返回栈顶元素
}

销毁栈 

void destory(SqStack* s){if(s->base){free(s->base);s->top=s->base=NULL;}
}

3.单调栈专题

从名字上我们就可以看出来,单调栈中存放的数据应该是有序的,所以单调栈本身也应该分为单调递增栈和单调递减栈。单调的方向是从栈底到栈顶。

在进行单调栈的讲解时,我们会使用到上述的一些基本操作。我们来看一下单调栈的入栈过程:

typedef int DataType;//此处选取int作为示例for (遍历这个数组)
{if (栈空 || 栈顶元素大于等于当前比较元素){入栈;}else{while (栈不为空 && 栈顶元素小于当前元素){栈顶元素出栈;更新结果;}当前数据入栈;}
}

真题演练

 柱状图中的最大矩形

思路:当前的数字可以向两边拓展,遇到比自己大的就接着拓展,小的就停止,然后用自己的高度乘以拓展的宽度,每次都更新最大面积,时间复杂度同样为O(N^2),所以我们接着借助单调栈

这里我们通过这道例题来使用一下单调递减栈

1.设置一个单调递减的栈(栈内0~n为单调递增)
2.当遇到小于栈顶元素的值,我们开始更新数据,因为有可能最大面积就会出现在栈中的序列里
3.牢记栈中数据永远是有序的,这个问题比较复杂,所以读者不妨对照着代码来理解问题

int largestRectangleArea(vector<int>& heights) {heights.push_back(-1);/同理,我们希望栈中所有数据出栈,所以给数组最后添加一个负数SqStack st; initStack(&st);int ret = 0, top;for (int i = 0; i < heights.size(); i++){if (isEmpty(st) || heights[Top(st)] <= heights[i]){Push(&st,i);}else{while (!isEmpty(st) && heights[Top(st)] > heights[i]){top = Top(st);Pop(&st);//i-top指的是当前矩形的宽度,heights[top]就是当前的高度//再次强调栈中现在为单调递增int tmp = (i - top)*heights[top];if (tmp > ret)ret = tmp;}Push(top);heights[top] = heights[i];}}return ret;
}

 4.共享栈专题

 利用栈底位置相对不变的特征,可让两个顺序栈共享一个一维数组空间,将两个栈的栈底分别设置在共享空间的两端,两个栈顶向共享空间的中间延伸,如下图所示:

两个栈共享同一片存储空间,这片存储空间不单独属于任何一个栈,某个栈需要的多一点,它就可能得到更多的存储空间;两个栈的栈底在这片存储空间的两端,当元素入栈时,两个栈的栈顶指针相向而行。此时我们采取另外一种实现栈的方法,不使用真正的指针,使用下标索引模拟两个栈顶指针。定义方法如下:

 #define SharedStackMax 100typedef char SharedStackType;typedef struct{SharedStackType data[SharedStackMax];size_t top1;size_t top2;}SharedStack;

共享栈的基本操作方法: 

 void SharedStackInit(SharedStack*stack){if(stack==NULL){return;}stack->top1=0;stack->top2=SharedStackMax;}void SharedStackPush1(SharedStack*stack,SharedStackType value){if(stack->top1==stack->top2||stack==NULL){return;}stack->data[stack->top1++]=value;return;}void SharedStackPush2(SharedStack*stack,SharedStackType value){if(stack->top2==stack->top1||stack==NULL){return;}                                                                       stack->data[--stack->top2]=value;}int  SharedStackTop1(SharedStack*stack,SharedStackType*value){if(stack==NULL||value==NULL){return 0;}if(stack->top1==0){return 0;}*value=stack->data[top1-1];return 1;}int SharedStackTop2(SharedStack*stack,SharedStackType*value){if(stack==NULL||value==NULL){return 0;}if(stack->top2==SharedStackMax){return 0;}*value=stack->data[stack->top2];return 1;
}void SharedStackPop1(SharedStack*stack){if(stack==NULL || stack->top1==0){return;}stack->top1--;}void SharedStackPop2(SharedStack*stack){if(stack->top2==SharedStackMax || stack==NULL){return;}stack->top2++;}     

链栈 

1.栈的链式存储

采用链式存储的栈称为链栈,链栈的优点是便于多个栈共享存储空间和提高其效率,且不存在栈满上溢的情况,我们就省去了不断扩容的麻烦。通常采用不带头结点的单链表实现,而且为了方便,使用头插法头删法更简单,所以我们的入栈和出栈就是单链表的头插和头删的操作。Lhead指向栈顶元素,链栈的结构图如下所示:

对于链栈来说,空栈的概念即是Lhead指向NULL的时候。 

/*构造节点*/
typedef struct StackNode{ElemType data;struct StackNode *next;
}StackNode, *LinkStackPrt;
//将StackNode* 重命名为LinkStackPrt的好处是避免下面操作出现二级指针的形式。(实质还是二级指针)/*构造链栈*/
typedef struct LinkStack{LinkStackPrt top;int count;
}LinkStack;

2.链栈的基本操作

链栈的进栈

对于链栈的进栈push操作,我们会创建一个栈节点,然后进行链表的头插操作。

void Push(LinkStack *S, ElemType e){LinkStackPrt p = (LinkStackPrt)malloc(sizeof(StackNode));p->data = e;p->next = S->top;    //把当前的栈顶元素赋值给新节点的直接后继S->top = p; //将新的结点S赋值给栈顶指针S->count++;
}

链栈的出栈

无需多说,头删即可

Element Pop(LinkStack *S){if(StackEmpty(*S)){return -1;}Element e = S->top->data;//临时存储栈顶元素LinkStackPtr p;p = S->top; //将栈顶结点赋值给pS->top = S->top->next;  //使得栈顶指针下移一位,指向后一结点free(p);    //释放结点pS->count--;return e;
}

栈的算法应用

递归专题

我们将递归时就提到过,递归是函数的递进调用与返回,而函数的调用与销毁又是以栈的形式进行的。所以能用栈解决的问题一定可以用递归解决。而能用递归的方式解决的问题,栈不一定能解决,但一定依托于栈的形式结构。因为对于栈结构本身来讲是作为一个容器来用的,递归才是存储一系列操作这种非数据类型的特殊栈结构。

基础算法--递归算法【难点、重点】-CSDN博客

调用堆栈:每次递归调用时,栈上会创建一个新的堆栈帧,用于存储该调用的上下文。

空间复杂度:递归的深度可能导致栈溢出,尤其在递归深度较大而没有适当的基本情况时。

迭代替代:有些递归算法可以用显式栈或循环来实现,从而避免递归带来的栈溢出问题。

四则运算表达式专题

四则运算在这里的应用其实包括后缀表达式和前缀表达式,以及我们平常熟悉的中缀表达式。

缀所处的位置不同代表的含义是四则运算符所处的位置不同。例如:

a+b*(c/d+2):中缀表达式

* + a b c:前缀表达式( ==(a+b)*c )

a b c d - * + e f / -:后缀表达式( ==a+b*(c-d)-e/f )

四则运算表达式的专题我们以一道例题作为训练:逆波兰表达式(前缀表达式)。

逆波兰表达式是一种把运算符前置的算术表达式,例如普通的表达式2 + 3的逆波兰表示法为+ 2 3。逆波兰表达式的优点是运算符之间不必有优先级关系,也不必用括号改变运算次序,例如(2 + 3) * 4的逆波兰表示法为* + 2 3 4。本题求解逆波兰表达式的值,其中运算符包括+ - * /四个。

对于原逆波兰表达式:- + * - c d b a  / e f

当我们遇到符号时,将其入栈,等到下面两个是数字时再出栈,如果遇到数字,直接让其入栈,然后出栈。我们对下面的图进行详细的步骤分析:

 

文字配合图片食用效果更甚哦:

下面给出代码: 

#include <bits/stdc++.h>
using namespace std;/*逆波兰表达式*/
string st = "+-*/";
double calc() {string str; cin >> str;switch (str[0]) {case '+':return calc() + calc();case '-':return calc() - calc();case '*':return calc() * calc();case '/':return calc() / calc();default:return stod(str);}
}
signed main() {printf("%f\n", calc());return 0;
}

感谢大家观看!多多支持哦!

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

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

五种IO模型与阻塞IO

一、前言 在网络中通信的本质其实是网络中的两台主机的进程间进行通信&#xff0c;而进程通信的本质就是IO。 IO分为输入&#xff08;input&#xff09;和输出&#xff08;output&#xff09;站在进程的角度讲&#xff0c;进程出去数据为输出&#xff0c;外部数据进入进程为输…

ubunut声卡配置 播放视频没有声音的解决方法 alsamixer和pavucontrol的使用方法

文章目录 &#x1f319;ubuntu22.04网页没有声音&#xff0c;声卡提示Dummy Output&#x1f319;方法一&#xff1a;切换内核&#x1f319;方法二&#xff1a;使用知乎的方法 &#x1f319;ubuntu22.04 连接蓝牙耳机&#xff0c;1秒后断连解决方法ubuntu声音操作alsamixerpavuc…

边缘计算插上AI的翅膀会咋样?

人工智能&#xff08;Artificial Intelligence,AI&#xff09;是研究、开发用于模拟、延伸和扩展人的智能的理论、方法、技术及应用系统的一门新的技术科学&#xff0c;是新一轮产业革命的重要驱动力量。2022年底发布的ChatGPT将人工智能技术上升到了一个新的高度。如今&#x…

17岁孩子开发AI应用,4个月入百万,人人都是AI产品经理的时代快来了

随着AI时代的到来叠加经济下行&#xff0c;越来越多的独立开发者梦想着实现年入百万的壮举。 近日&#xff0c;这种小概率事件正在发生。 17岁高中生做了个AI APP&#xff0c;短短四个月销售额达100 万美元。 小伙儿Zach Yadegari&#xff08;下面暂称小扎克&#xff09;在X…

用IMX6UL开发板编写按键输入实验

在之前我们都是讲解如何使用IMX6UL的GPIO输出控制等功能&#xff0c;IMX6U的IO不仅能作为输出&#xff0c;而且也可以作为输入&#xff0c;而我们开发板上具有一个按键&#xff0c;按键肯定是连接了一个IO口的额&#xff0c;我们在这一节将会把IO配置成输入功能&#xff0c;读取…

codetop标签动态规划大全C++讲解(三)!!动态规划刷穿地心!!学吐了家人们o(╥﹏╥)o

每天复习一篇&#xff0c;只有十题左右 1.买卖股票的最佳时机2.买卖股票的最佳时机含手续费3.买卖股票的最佳时机III4.买卖股票的最佳时机IV5.打家劫舍6.打家劫舍II7.不同路径8.不同路径II9.最小路径和10.三角形的最小路径和11.两个字符串的删除操作12.编辑距离13.一和零 1.买卖…

强化学习笔记之【DDPG算法】

强化学习笔记之【DDPG算法】 文章目录 强化学习笔记之【DDPG算法】前言&#xff1a;原论文伪代码DDPG算法DDPG 中的四个网络代码核心更新公式 前言&#xff1a; 本文为强化学习笔记第二篇&#xff0c;第一篇讲的是Q-learning和DQN 就是因为DDPG引入了Actor-Critic模型&#x…

Ubuntu22.04 Docker 国内安装最靠谱教程

目前docker在国内安装常存在众所周知的网络问题&#xff0c;如果安装过程如果从官网地址安装以及安装之后从官网要拉取镜像都存在问题。这篇文章主要针对这两个问题总结最靠谱的docker安装教程。 1. docker安装 1.1 系统环境概述 Ubuntu 22.04linux内核版本 6.8&#xff08;…

重学SpringBoot3-集成Redis(四)之Redisson

更多SpringBoot3内容请关注我的专栏&#xff1a;《SpringBoot3》 期待您的点赞&#x1f44d;收藏⭐评论✍ 重学SpringBoot3-集成Redis&#xff08;四&#xff09;之Redisson 1. 添加 Redisson 依赖2. 配置 Redisson 客户端3. 使用 Redisson 实现分布式锁4. 调用分布式锁5. 为什…

二进制的神奇操作——拆位法和贡献思想

拆位的引入 我们来思考这么一个问题&#xff0c;如果给你一个数组&#xff0c;让你去求一个数组里面所有连续子串的异或和的和&#xff0c;问你该怎么求&#xff1f; 我们该如何去处理&#xff0c;首先肯定是会想到暴力的思路&#xff0c;第一层循环遍历左端点&#xff0c;第…

SpringBoot在线教育平台:设计与实现的深度解析

2相关技术 2.1 MYSQL数据库 MySQL是一个真正的多用户、多线程SQL数据库服务器。 是基于SQL的客户/服务器模式的关系数据库管理系统&#xff0c;它的有点有有功能强大、使用简单、管理方便、安全可靠性高、运行速度快、多线程、跨平台性、完全网络化、稳定性等&#xff0c;非常…

654、最大二叉树

1、题目描述 . - 力扣&#xff08;LeetCode&#xff09; 其实就是给定了一个所谓"最大二叉树"的规则&#xff0c;让我们去构建二叉树。 以 nums [3,2,1,6,0,5] 为例&#xff0c;规则如下&#xff1a; (1)找出其中的最大值6将其作为根节点&#xff0c;6前面的是左子…

程序传入单片机的过程,以Avrdude为例分析

在市场上有各式各样的单片机&#xff0c;例如Arduino&#xff0c;51单片机&#xff0c;STM等。通常&#xff0c;我们都用其对应的IDE软件进行单片机的编程。这些软件既负责将程序代码转写成二进制代码&#xff0c;即机器语言&#xff0c;也负责将该二进制代码导入单片机。与此同…

C++ 算法学习——7.4.1 优化算法——双指针

双指针法&#xff08;Two Pointers&#xff09;是一种常用的算法技巧&#xff0c;通常用于解决数组或链表中的问题。这种技巧通过维护两个指针&#xff0c;通常分别指向数组或链表的不同位置&#xff0c;来协同解决问题。双指针法一般有两种类型&#xff1a;快慢指针和左右指针…

什么是transformer大模型,答案就在这里

Transformer大模型是一种在自然语言处理&#xff08;NLP&#xff09;领域中广泛使用的模型&#xff0c;其详细数据与分析可以从以下几个方面进行阐述&#xff1a; 1. 模型架构 Transformer模型本质上是一个Encoder-Decoder架构。编码组件由多层编码器&#xff08;Encoder&…

(笔记)第三期书生·浦语大模型实战营(十一卷王场)–书生基础岛第3关---浦语提示词工程实践

学员闯关手册&#xff1a;https://aicarrier.feishu.cn/wiki/ZcgkwqteZi9s4ZkYr0Gcayg1n1g?open_in_browsertrue 课程视频&#xff1a;https://www.bilibili.com/video/BV1cU411S7iV/ 课程文档&#xff1a; https://github.com/InternLM/Tutorial/tree/camp3/docs/L1/Prompt 关…

还在“卷”长度?长文本模型真的基于上下文进行回复吗?

近年来&#xff0c;随着长文本模型&#xff08;Long-context Model, LCM&#xff09;技术的突飞猛进&#xff0c;处理长上下文的能力已成为各大语言模型&#xff08;Large Language Model, LLM&#xff09;的核心竞争力&#xff0c;也是各大技术厂商争夺的焦点。截至2023年12月…

RAG再总结之如何使大模型更好使用外部数据:四个不同层级及查询-文档对齐策略

我们来看看RAG进展。《Retrieval Augmented Generation (RAG) and Beyond: A Comprehensive Survey on How to Make your LLMs use External Data More Wisely》(https://arxiv.org/abs/2409.14924)&#xff0c;主要讨论了如何使大型语言模型&#xff08;LLMs&#xff09;更明智…

Redis中BitMap实现签到与统计连续签到功能

服务层代码 //签到Overridepublic Result sign() {//1.获取当前登录的用户Long userId UserHolder.getUser().getId();//获取日期LocalDateTime now LocalDateTime.now();//拼接keyString keySuffix now.format(DateTimeFormatter.ofPattern(":yyyyMM"));String …

实例分割、语义分割和 SAM(Segment Anything Model)

实例分割、语义分割和 SAM&#xff08;Segment Anything Model&#xff09; 都是图像处理中的重要技术&#xff0c;它们的目标是通过分割图像中的不同对象或区域来帮助识别和分析图像&#xff0c;但它们的工作方式和适用场景各有不同。 1. 语义分割&#xff08;Semantic Segme…