数据结构与算法(C/C++版)【栈与队列】
第三章《栈与队列》
(一)栈简介
栈(Stack):只允许在一端进行插入或删除操作的线性表。首先栈是一种线性表,但是限定这种线性表只能在某一端进行插入和删除操作
栈顶(top):线性表允许进行插入和删除的那一端。(开口的那一端)
栈底(bottom):固定的,不允许进行插入和删除的另一端。(封死的那一端)
空栈:不含任何元素的空表。
栈的“先进后出”原则(FILO):已上图为例,栈中存放了 4 个数据元素,进栈的顺序是 A 先进栈,然后 B 进,然后 C 进,最后 D 进栈;当需要调取 A 时,首先 D 出栈,然后 C 出栈,然后 B 出栈,最后 A 才能出栈被调用。
栈的两种表示方式:栈的本质是线性表,那么它就同样有线性表的两种表示形式:顺序栈 和 链式栈(简称“链栈”)
两者的区别:存储的数据元素在物理结构上是否是相互紧挨着的。顺序栈存储元素预先申请连续的存储单元;链栈需要即申请,数据元素不紧挨着。
栈的“上溢”和“下溢”问题:
“上溢”:在栈已经存满数据元素的情况下,如果继续向栈内存入数据,栈存储就会出错。(栈满还存会“上溢”)
“下溢”:在栈内为空的状态下,如果对栈继续进行取数据的操作,就会出错。(栈空再取会“下溢”)
对于栈的两种表示方式来说,顺序栈两种情况都有可能发生;而链栈由于“随时需要,随时申请空间”的存储结构,不会出现“上溢”的情况。
栈的基本操作:
InitStack(&S):初始化一个空栈S。
StackEmpty(S):判断一个栈是否为空,若栈S为空返回true,否则返回false。
Push(&S, x):进栈,若栈S未满,将x加入使之成为新桟顶。
Pop(&S, &x):出栈,若栈S非空,弹出栈顶元素,并用x返回。
GetTop(S, &x):读栈顶元素,若栈S非空,用x返回栈顶元素。
ClearStack(&S):销毁栈,并释放栈S占用的存储空间。
注意:符号'&'是C++特有的,用来表示引用,有的书上釆用C语言中的指针类型‘*’,也可以达到传址的目的。
(二)顺序栈
1、简介
定义:栈的顺序存储称为顺序栈,它是利用一组地址连续的存储单元存放自栈底到栈顶的数据元素,同时附设一个指针(top)指示当前栈顶的位置。
(a)是空栈,(c)是A、B、C、D、E共5 个元素依次入栈后的结果,(d)是在图3-2 (c)之后E、D、C相继出栈,此时栈中还 有2个元素,或许最近出栈的元素C、D、E仍在原先的单元存储着,但top指针已经指向了 新的栈顶,则元素C、D、E已不在栈中了,
栈顶指针:S.top,初始时设置S.top=-1;栈顶元素:S.data[S.top]。
进栈操作:栈不满时,栈顶指针先加1,再送值到栈顶元素。
出栈操作:栈非空时,先取栈顶元素值,再将栈顶指针减1。
栈空条件:S.top=-1;栈满条件:S.top==MaxSize-1;栈长:S.top+1。
2、常见操作
① 结构体定义
#define MaxSize 50 //定义栈中元素的最大个数
typedef struct{
Elemtype data[MaxSize]; //存放栈中元素
int top; //栈顶指针
}SqStack;
② 初始化
void initStack(Sqstack &st)
{
st.top=-; //初始化栈顶指针
}
③ 判断是否为空
void isEmpty(Sqstack st)
{
if(st.top==-) //栈空
return ;
else //不空
return ;
}
④ 进栈
int push(Sqstack &st,int &x)
{
if(st.top==maxSize-) //栈满,报错
return ;
++(st.top); //指针先加 1(先移动指针),再进栈
st.data[st.top]=x;
return ;
}
⑤ 出栈
int pop(Sqstack &st,int &x)
{
if(st.top==-) //栈空,报错
return ;
x=st.data[st.top]; //先出栈,指针再减1(先取出元素,在移动指针)
--(st.top);
return ;
}
⑥ 读栈顶元素
bool GetTop(SqStack S,ElemType &x){
if (S.top==-) //找空,报错
return false;
x=S.data[S.top]; //x记录栈顶元素
return true;
}
⑦ 简化写法
//1.初始化
int stack[maxSize];
int top=-;
//2.进栈
stack[++top]=x;
//3.出栈
x=stack[--top];
3、共享栈
利用栈底位置相对不变的特性,可以让两个顺序栈共享一个一维数据空间,将两个栈的 栈底分别设置在共享空间的两端,两个栈顶向共享空间的中间延伸。
两个栈的栈顶指针都指向栈顶元素,top0=-1 时0号桟为空,top1=MaxSize时1号栈为空;仅当两个栈顶指针相邻(top1-top0=1) 时,判断为栈满。当0号栈进栈时top0先加1 再赋值,1号栈进栈时top1先减1再赋值;出栈时则刚好相反。
共享栈是为了更有效地利用存储空间,两个栈的空间相互调节,只有在整个存储空间被 占满时才发生上溢。其存取数据的时间复杂度均为0(1),所以对存取效率没有什么影响。
(三)链栈
釆用链式存储的栈称为链栈,链栈的优点是便于多个栈共享存储空间和提高其效率,且 不存在栈满上溢的情况。通常釆用单链表实现,并规定所有操作都是在单链表的表头进行的。 这里规定链栈没有头结点(链栈一般不需要创建头结点,头结点会增加程序的复杂性,只需要创建一个头指针就可以了),Lhead指向栈顶元素。
1、常见操作
① 初始化
void initStack(LNode*&lst) //lst要改变,用引用型
{
lst=(LNode*)malloc(sizeof(LNode)); //创建一个头结点
lst->next=NULL; }
② 判断是否为空
void isEmpty(Sqstack *lst)
{
if(lst->next==NULL) //栈空
return ;
else //不空
return ;
}
③ 进栈
int push(Sqstack *lst,int x)
{
LNode *p;
p=(LNode*)malloc(sizeof(LNode)); //为进栈元素申请结点空间(默认内存无限大)
p->next=NULL; //初始化(没新申请一个结点,将其指针域设置为NULL) //以下是单链表的头插法
p->next =x;
p->next =lst->next;
lst->next=p;
}
④ 出栈
int pop(Sqstack *lst,int &x)
{
LNode *p;
if(lst->next==NULL) //栈空不能出栈
return=; //以下是单链表的删除操作
p=(LNode*)malloc(sizeof(LNode)); //为进栈元素申请结点空间
p->next=NULL; //初始化(没新申请一个结点,将其指针域设置为NULL) //以下是链表的头插法
p = lst->next;
x=p->data;
lst->next=p->next;
free(p);
return ;
}
(四)队列简介
队列(Queue):队列简称队,也是一种操作受限的线性表,只允许在表的一端进行插入,而在表的另一端进行删除。
“先进先出”(FIFO):这和我们日常生活中的排队是一致的,最早排队的也是最早离队的。
队头(Front):允许删除的一端,又称为队首。
队尾(Rear):允许插入的一端。
空队列:不含任何元素的空表。
队列常见的基本操作:
InitQueue(&Q):初始化队列,构造一个空队列Q。
QueueEmpty(Q):判队列空,若队列Q为空返回true,否则返回false。
EnQueue(&Q, x):入队,若队列Q未满,将x加入,使之成为新的队尾。
DeQueue(&Q, &x):出队,若队列Q非空,删除队头元素,并用x返回。
GetHead(Q, &x):读队头元素,若队列Q非空,则将队头元素赋值给X。
需要注意的是,队列是操作受限的线性表,所以,不是任何对线性表的操作都可以作为队列的操作。比如,不可以随便读取队列中间的某个数据。
(五)顺序队列、循环队列
<1>顺序队列
队列的顺序实现是指分配一块连续的存储单元存放队列中的元素,并附设两个指针front 和rear分别指示队头元素和队尾元素的位置。设队头指针指向队头元素,队尾指针指向队尾 元素的下一个位置(也可以让rear指向队尾元素,front指向队头元素的前一个位置)
结构体定义:
#define MaxSize 50 //定义队列中元素的最大个数
typedef struct{
ElemType data[MaxSize]; //存放队歹I]元素
int front, rear; //队头指针和队尾指针
}SqQueue;
初始状态(队空条件):Q.front==Q.rear==0。
进队操作:队不满时,先送值到队尾元素,再将队尾指针加1。
出队操作:队不空时,先取队头元素值,再将队头指针加1。
(a)所示为队列的初始状态,有Q.front==Q.rear==0成立,该条件可以作为队列判空的条件。但能否用Q.rear==MaxSize作为队列满的条件呢?显然不能,图3-6(d)中, 队列中仅有1个元素,但仍满足该条件。这时入队出现“上溢出”,但这种溢出并不是真正 的溢出,在data数组中依然存在可以存放元素的空位置,所以是一种“假溢出”。
<2>循环队列
为了解决顺序队列的缺点将顺序队列变为一个环状的空间,即把存储队列元素的表从逻辑上看成一个环,称为循环队列。当队首指针Q.ftont =MaxSiZe-1后,再前进一个位置就自动到0,这可以利用除法取余运算(%)来实现。
初始时:Q.front=Q.rear=0
队首指针进 1:Q.front=(Q.front+1)%MaxSize
队尾指针进 1:Q.rear=(Q.rear+1)%MaxSize
队列长度:(Q.rear+MaxSize-Q.front)%MaxSize
出队入队时:指针都按顺时针方向进1
那么,循环队列队空和队满的判断条件是什么呢?显然,队空的条件是Q.front==Q.rear。 如果入队元素的速度快于出队元素的速度,队尾指针很快就赶上了队首指针,如图(d1),此时可以看出队满时也有Q.front==Q.rear。
为了区分队空还是队满的情况,有三种处理方式:
1) 牺牲一个单元来区分队空和队满,入队时少用一个队列单元,这是一种较为普遍的做法,约定以“队头指针在队尾指针的下一位置作为队满的标志”,如图(d2)所示。
队满条件为: (Q.rearfl)%MaxSize==Q.front。
队空条件仍为:Q.front==Q.rear。
队列中元素的个数:(Q.rear-Q.front+MaxSize)%MaxSize
2) 类型中增设表示元素个数的数据成员。这样,则队空的条件为Q.Size==0,队满的条 件为 Q.size==MaxSize。这两种情况都有 Q.front=Q.rear。
3) 类型中增设tag数据成员,以区分是队满还是队空。tag等于0的情况下,若因删除导致Q.front==Q.rear则为队空;tag等于1的情况下,若因插入导致Q.ftont==Q.rear则为队满。
循环队列的操作:
①初始化
void InitQueue(&Q)
{
Q.rear=Q.front=; //初始化队首、队尾指针
}
②判队空
bool isEmpty(Q)
{
if(Q.rear == Q.front) //队空条件
return true;
else
return false;
}
③入队
bool EnQueue(SqQueue &Q, ElemType x)
{
if((Q.rear+)%MaxSize == Q.front) //队满
return false;
Q.rear= (Q.rear+)%MaxSize; //队未满(先移动指针)
Q.data[Q.rear]=x; //(再存入数据)
return true;
}
④出队
bool DeQueue(SqQueue &Q, ElemType &x)
{
if(Q.rear == Q.front)
return false; //队空,报错
Q.front= (Q.front+)%MaxSize; //队不空(先移动指针)
x=Q.data[Q.front]; //(再存入数据)
return true;
}
(六)链队
队列的链式表示称为链队列,它实际上是一个同时带有队头指针和队尾指针的单链表。头指针指向队头结点,尾指针指向队尾结点,即单链表的最后一个结点。
不设头结点的链式队列在操作上往往比较麻烦,因此,通常将链式队列设计 成一个带头结点的单链表,这样插入和删除操作就统一了。
用单链表表示的链式队列特别适合于数据元素变动比较大的情形,而且不存在队列满且产生溢出的问题。另外, 假如程序中要使用多个队列,与多个栈的情形一样,最好使用链式队列,这样就不会出现存储分配不合理和“溢出” 的问题。
链式队列的基本操作(跟线性表类似):
① 初始化
void InitQueue(LinkQueue &Q)
{
Q.front = Q.rear=(LinkNode*)malloc(sizeof(LinkNode)); //创建头结点
Q.front->next=NULL; //初始为空
}
② 判队空
bool IsEmpty(LinkQueue Q)
{
if(Q.front==Q.rear)
return true;
else
return false;
}
③入队
③入队
void EnQueue(LinkQueue &Q, ElemType x)
{
s=(LinkNode *)malloc(sizeof(LinkNode));//创建新结点
s->data=x;
s->next=NULL;
Q.rear->next=s; //插入到链尾
Q.rear=s;
}
④出队
bool DeQueue(LinkQueue &Q, ElemType &x)
{
if (Q.front == Q.rear)
return false; //空队
p=Q.front->next;
x=p->data;
Q.front->next=p->next;
if(Q.rear==p)
Q.rear=Q.front; //若原队列中只有一个结点,删除后变空
free(p);
return true;
}
(七)双端队列
双端队列是指允许两端都可以进行入队和出队操作的队列。其元素的 逻辑结构仍是线性结构。将队列的两端分别称为前端和后端,两端都可以入队和出队。
进队时:前端进的元素排列在队列中后端进的元素的前面,后端进的元素排列在队列中前端进的元素的后面。
出队时:无论前端还是后端出队,先出的元素排列在后出的元素的前面。
①输出受限的双端队列:允许在一端进行插入和删除,但在另一端只允许插入的双端队列。
②输入受限的双端队列:允许在一端进行插入和删除,但在另一端只允许删除的双端队列。而如果限定双端队列从某个端点插入的元素只 能从该端点删除,则该双端队列就蜕变为两个栈底相邻接的栈了。
(八)栈的应用:
①括号匹配
②表达式求值
③递归
(九)队列的应用:
①层次遍历
②解决主机与外部设备之间速度不匹配问题
③解决由多用户引起的资源竞争问题
关于应用以后有时间回来补充。。。。
数据结构与算法(C/C++版)【栈与队列】的更多相关文章
- 重读《学习JavaScript数据结构与算法-第三版》- 第4章 栈
定场诗 金山竹影几千秋,云索高飞水自流: 万里长江飘玉带,一轮银月滚金球. 远自湖北三千里,近到江南十六州: 美景一时观不透,天缘有分画中游. 前言 本章是重读<学习JavaScript数据结构 ...
- 重读《学习JavaScript数据结构与算法-第三版》- 第5章 队列
定场诗 马瘦毛长蹄子肥,儿子偷爹不算贼,瞎大爷娶个瞎大奶奶,老两口过了多半辈,谁也没看见谁! 前言 本章为重读<学习JavaScript数据结构与算法-第三版>的系列文章,主要讲述队列数据 ...
- 图解堆算法、链表、栈与队列(Mark)
原文地址: 图解堆算法.链表.栈与队列(多图预警) 堆(heap),是一类特殊的数据结构的统称.它通常被看作一棵树的数组对象.在队列中,调度程序反复提取队列中的第一个作业并运行,因为实际情况中某些时间 ...
- JavaScript 版数据结构与算法(一)栈
今天,我们要讲的是数据结构与算法中的栈. 栈的简介 栈是什么?栈是一个后进先出(LIFO)的数据结构.栈有啥作用?栈可以模拟算法或生活中的一些后进先出的场景,比如: 十进制转二进制,你需要将余数倒序输 ...
- Java数据结构和算法(四)——栈
前面我们讲解了数组,数组更多的是用来进行数据的存储,纯粹用来存储数据的数据结构,我们期望的是插入.删除和查找性能都比较好.对于无序数组,插入快,但是删除和查找都很慢,为了解决这些问题,后面我们会讲解比 ...
- 数据结构1:数据结构与算法C语言版分析概述
本节开始将带领大家系统地学习数据结构,作为一门计算机专业大二学生的必修课程,该课程面对的目标人群为初步具备基本编程能力和编程思想的程序员(大一接触了 C 语言或者 C++).通过系统地学习数据结构,可 ...
- 重读《学习JavaScript数据结构与算法-第三版》-第2章 ECMAScript与TypeScript概述
定场诗 八月中秋白露,路上行人凄凉: 小桥流水桂花香,日夜千思万想. 心中不得宁静,清早览罢文章, 十年寒苦在书房,方显才高志广. 前言 洛伊安妮·格罗纳女士所著的<学习JavaScript数据 ...
- JavaScript 数据结构与算法1(数组与栈)
学习数据结构的 git 代码地址: https://gitee.com/zhangning187/js-data-structure-study 1.数组 几乎所有的语言都原生支持数组类型,因为数组是 ...
- 重读《学习JavaScript数据结构与算法-第三版》- 第3章 数组(一)
定场诗 大将生来胆气豪,腰横秋水雁翎刀. 风吹鼍鼓山河动,电闪旌旗日月高. 天上麒麟原有种,穴中蝼蚁岂能逃. 太平待诏归来日,朕与先生解战袍. 此处应该有掌声... 前言 读<学习JavaScr ...
- 重读《学习JavaScript数据结构与算法-第三版》- 第6章 链表(一)
定场诗 伤情最是晚凉天,憔悴厮人不堪言: 邀酒摧肠三杯醉.寻香惊梦五更寒. 钗头凤斜卿有泪,荼蘼花了我无缘: 小楼寂寞新雨月.也难如钩也难圆. 前言 本章为重读<学习JavaScript数据结构 ...
随机推荐
- A Walk Through the Forest
A Walk Through the Forest Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/O ...
- Android 开发笔记___登陆app
package com.example.alimjan.hello_world; /** * Created by alimjan on 7/4/2017. */ import android.con ...
- SpringBoot下配置FreeMarker配置远程模版
需求产生原因 要求在同一个接口中,根据不同的参数,返回不同的视图结果 所有的视图中的数据基本一致 要求页面能静态化,优化SEO 例如:A接口返回客户的信息 客户A在调用接口时,返回其个性化定制的页面A ...
- Python爬虫入门:综述
大家好哈,最近博主在学习Python,学习期间也遇到一些问题,获得了一些经验,在此将自己的学习系统地整理下来,如果大家有兴趣学习爬虫的话,可以将这些文章作为参考,也欢迎大家一共分享学习经验. Pyth ...
- Java IO编程全解(五)——AIO编程
转载请注明出处:http://www.cnblogs.com/Joanna-Yan/p/7794151.html 前面讲到:Java IO编程全解(四)--NIO编程 NIO2.0引入了新的异步通道的 ...
- SQL Server分组查询某最大值的整条数据(包含linq写法)
想实现如下效果,就是分组后时间最大的那一条数据: 1.SQL SELECT * FROM ( SELECT * , ROW_NUMBER() OVER ( PARTITION BY RIP_GUID ...
- 项目实战2—LVS负载均衡
负载均衡集群企业级应用实战-LVS 实现基于LVS负载均衡集群的电商网站架构 随着业务的发展,网站的访问量越来越大,网站访问量已经从原来的1000QPS,变为3000QPS,网站已经不堪重负,响应缓慢 ...
- redis源码分析之事务Transaction(上)
这周学习了一下redis事务功能的实现原理,本来是想用一篇文章进行总结的,写完以后发现这块内容比较多,而且多个命令之间又互相依赖,放在一篇文章里一方面篇幅会比较大,另一方面文章组织结构会比较乱,不容易 ...
- php 文档操作
ftp_mkdir() 函数在 FTP 服务器上建立新目录. 语法 ftp_mkdir(ftp_connection,dir) 参数 描述 ftp_connection 必需.规定要使用的 FTP 连 ...
- Nodejs前端服务器压缩图片
Nodejs作为前端服务器,自然能承担处理图片的能力, 使用GM for nodejs 作为图片处理器,调用ImageMagick处理图片 使用ImageMagick var imageMagick ...