3.1 栈和队列的定义和特点
栈


队列 
3.2 栈、队列与一般线性表的区别

3.3 栈的表示和操作的实现
顺序栈与顺序表

=================
顺序栈的表示

#define MAXSIZE 100
typedef struct
{
SElemType *base;
SElemType *top;
int stacksize;
}SqStack;
顺序栈初始化

Status InitStack( SqStack &S )
{
S.base =new SElemType[MAXSIZE];
if( !S.base ) return OVERFLOW;
S.top = S.base;
S.stackSize = MAXSIZE;
return OK;
}
判断顺序栈是否为空
bool StackEmpty( SqStack S )
{
if(S.top == S.base) return true;
else return false;
}
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-us7z0ZZA-1634984533079)(C:\Users\Hasee\AppData\Roaming\Typora\typora-user-images\image-20211023113807456.png)]](https://img-blog.csdnimg.cn/7f16026eccc0467dbe22e60a47b2a971.png)
求顺序栈的长度
int StackLength( SqStack S )
{
return S.top – S.base;
}
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-SueartJJ-1634984533080)(C:\Users\Hasee\AppData\Roaming\Typora\typora-user-images\image-20211023113816454.png)]](https://img-blog.csdnimg.cn/bd8920773cdc47e08596e2f186cb6aa6.png)
清空顺序栈
Status ClearStack( SqStack S )
{
if( S.base ) S.top = S.base;
return OK;
}

销毁顺序栈
Status DestroyStack( SqStack &S )
{
if( S.base )
{
delete S.base ;
S.stacksize = 0;
S.base = S.top = NULL;
}
return OK;
}
顺序栈进栈
(1)判断是否栈满,若满则出错 (2)元素e压入栈顶 (3)栈顶指针加1
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-bH86RR2l-1634984533082)(C:\Users\Hasee\AppData\Roaming\Typora\typora-user-images\image-20211023114204502.png)]](https://img-blog.csdnimg.cn/30784db86305401f955f36164a837af3.png)
Status Push( SqStack &S, SElemType e)
{
if( S.top - S.base== S.stacksize )
return ERROR;
*S.top++=e;
return OK;
}

顺序栈出栈
(1)判断是否栈空,若空则出错 (2)获取栈顶元素e (3)栈顶指针减1
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-HRo6ONbp-1634984533084)(C:\Users\Hasee\AppData\Roaming\Typora\typora-user-images\image-20211023114307320.png)]](https://img-blog.csdnimg.cn/1de72d15e8ff4b0093fe104400f7af33.png)
Status Pop( SqStack &S, SElemType &e)
{
if( S.top == S.base )
return ERROR;
e= *--S.top;
return OK;
}
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-VXXeoR7G-1634984533085)(C:\Users\Hasee\AppData\Roaming\Typora\typora-user-images\image-20211023114339343.png)]](https://img-blog.csdnimg.cn/a59c53dbf0f64b5e93d9512b856af137.png)
取顺序栈栈顶元素
- 判断是否空栈,若空则返回错误
- 否则通过栈顶指针获取栈顶元素
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-n00pJy4A-1634984533085)(C:\Users\Hasee\AppData\Roaming\Typora\typora-user-images\image-20211023114742535.png)]](https://img-blog.csdnimg.cn/17d09a7115a34544ae3c2872e896c5fe.png)
Status GetTop( SqStack S, SElemType &e)
{
if( S.top == S.base ) return ERROR;
e = *( S.top – 1 );
return OK;
}
==================
链栈的表示
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-gcqUCMoP-1634984533086)(C:\Users\Hasee\AppData\Roaming\Typora\typora-user-images\image-20211023165432412.png)]](https://img-blog.csdnimg.cn/e45aea55fbf14bc9868fa42bfcf0c589.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBA5q-bX-S4ieaciA==,size_14,color_FFFFFF,t_70,g_se,x_16)
typedef struct StackNode {
SElemType data;
struct StackNode *next;
} StackNode, *LinkStack;
LinkStack S;
链栈的初始化
void InitStack(LinkStack &S )
{
S=NULL;
}
判断链栈是否为空
Status StackEmpty(LinkStack S)
{
if (S==NULL) return TRUE;
else return FALSE;
}
链栈进栈
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-fvdAdwrx-1634984533087)(C:\Users\Hasee\AppData\Roaming\Typora\typora-user-images\image-20211023165841929.png)]](https://img-blog.csdnimg.cn/b6b353286e0341999c6920fc335b8cd8.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBA5q-bX-S4ieaciA==,size_15,color_FFFFFF,t_70,g_se,x_16)
Status Push(LinkStack &S , SElemType e)
{
p=new StackNode;
if (!p) exit(OVERFLOW);
p->data=e; p->next=S; S=p;
return OK; }
链栈出栈
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-PBhGQ1Jw-1634984533087)(C:\Users\Hasee\AppData\Roaming\Typora\typora-user-images\image-20211023170251679.png)]](https://img-blog.csdnimg.cn/7a2efa1a0ceb4f418c380d5bb4dce34f.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBA5q-bX-S4ieaciA==,size_15,color_FFFFFF,t_70,g_se,x_16)
Status Pop (LinkStack &S,SElemType &e)
{if (S==NULL) return ERROR;
e = S-> data; p = S; S = S-> next;
delete p; return OK; }
取链栈栈顶元素
SElemType GetTop(LinkStack S)
{
if (S==NULL) exit(1); else return S–>data;
}
3.4 栈与递归
递归的定义
若一个对象部分地包含它自己, 或用它自己给自己定义, 则称这个对象是递归的;
若一个过程直接地或间接地调用自己, 则称这个过程是递归的过程。
long Fact ( long n ) {
if ( n == 0) return 1;
else return n * Fact (n-1); }
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-5nPHmtdc-1634984533088)(C:\Users\Hasee\AppData\Roaming\Typora\typora-user-images\image-20211023170643945.png)]](https://img-blog.csdnimg.cn/9dc5bdccf7184c66a01100f4637a5b46.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBA5q-bX-S4ieaciA==,size_20,color_FFFFFF,t_70,g_se,x_16)
以下三种情况常常用到递归方法
- 递归定义的数学函数
- 具有递归特性的数据结构
- 可递归求解的问题
1. 递归定义的数学函数:
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-4fI3KrUm-1634984533088)(C:\Users\Hasee\AppData\Roaming\Typora\typora-user-images\image-20211023170857799.png)]](https://img-blog.csdnimg.cn/1e5774aec8cb459297b10e7b933223ad.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBA5q-bX-S4ieaciA==,size_20,color_FFFFFF,t_70,g_se,x_16)
2. 具有递归特性的数据结构:
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-9wJ9nhJa-1634984533089)(C:\Users\Hasee\AppData\Roaming\Typora\typora-user-images\image-20211023173450960.png)]](https://img-blog.csdnimg.cn/1c1beb813759484e8c05e83d5ecee80c.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBA5q-bX-S4ieaciA==,size_20,color_FFFFFF,t_70,g_se,x_16)
3. 可递归求解的问题:
递归算法的效率分析
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-x5x1fL5N-1634984533090)(C:\Users\Hasee\AppData\Roaming\Typora\typora-user-images\image-20211023173731133.png)]](https://img-blog.csdnimg.cn/292f2dc87305464f9357b4a8c09b26ba.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBA5q-bX-S4ieaciA==,size_20,color_FFFFFF,t_70,g_se,x_16)
递归的优缺点
优点:结构清晰,程序易读
缺点:每次调用要生成工作记录,保存状态信息,入栈;返回时要出栈,恢复状态信息。时间开销大。
3.5 队列的表示和操作的实现
队列的抽象数据类型
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-MJ9XZJoJ-1634984533091)(C:\Users\Hasee\AppData\Roaming\Typora\typora-user-images\image-20211023174511921.png)]](https://img-blog.csdnimg.cn/04f930c7acc346eca8d05b2e1c34a19a.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBA5q-bX-S4ieaciA==,size_20,color_FFFFFF,t_70,g_se,x_16)
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-tKQ9ifGK-1634984533092)(C:\Users\Hasee\AppData\Roaming\Typora\typora-user-images\image-20211023174524756.png)]](https://img-blog.csdnimg.cn/d2b476819d224efa9ef932233b1fd315.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBA5q-bX-S4ieaciA==,size_20,color_FFFFFF,t_70,g_se,x_16)
队列的顺序表示--用一维数组base[M]
#define M 100
Typedef struct {
QElemType *base;
int front;
int rear;
}SqQueue;
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-gtP2QPqk-1634984533092)(C:\Users\Hasee\AppData\Roaming\Typora\typora-user-images\image-20211023174649338.png)]](https://img-blog.csdnimg.cn/c5e9b86bf76d471592de15667bc3bf34.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBA5q-bX-S4ieaciA==,size_20,color_FFFFFF,t_70,g_se,x_16)
存在的问题 设大小为M
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-36nwKejQ-1634984533093)(C:\Users\Hasee\AppData\Roaming\Typora\typora-user-images\image-20211023174924363.png)]](https://img-blog.csdnimg.cn/79404aa569ce4f2b836a9c85e5bbd1be.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBA5q-bX-S4ieaciA==,size_20,color_FFFFFF,t_70,g_se,x_16)
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-mysYEdJa-1634984533093)(C:\Users\Hasee\AppData\Roaming\Typora\typora-user-images\image-20211023175009158.png)]](https://img-blog.csdnimg.cn/f44ef481edfd44289f93fbc4e2e6848c.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBA5q-bX-S4ieaciA==,size_20,color_FFFFFF,t_70,g_se,x_16)
解决的方法--循环队列
base[0]接在base[M-1]之后
若rear+1==M
则令rear=0;
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-iPRu6lZC-1634984533094)(C:\Users\Hasee\AppData\Roaming\Typora\typora-user-images\image-20211023175120996.png)]](https://img-blog.csdnimg.cn/d8f2891286304109bdc0f2580857ee91.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBA5q-bX-S4ieaciA==,size_19,color_FFFFFF,t_70,g_se,x_16)
实现:利用“模”运算
入队:
base[rear]=x;
rear=(rear+1)%M;
出队:
x=base[front];
front=(front+1)%M;
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-YGLwEGdG-1634984533095)(C:\Users\Hasee\AppData\Roaming\Typora\typora-user-images\image-20211023175205526.png)]](https://img-blog.csdnimg.cn/7494228e683f40b3adf65b38226d696d.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBA5q-bX-S4ieaciA==,size_17,color_FFFFFF,t_70,g_se,x_16)
循环队列
#define MAXQSIZE 100
Typedef struct {
QElemType *base;
int front;
int rear;
}SqQueue;
循环队列初始化
Status InitQueue (SqQueue &Q){
Q.base =new QElemType[MAXQSIZE]
if(!Q.base) exit(OVERFLOW);
Q.front=Q.rear=0;
return OK;
}
求循环队列的长度
int QueueLength (SqQueue Q){
return (Q.rear-Q.front+MAXQSIZE)%MAXQSIZE;
}
循环队列入队
Status EnQueue(SqQueue &Q,QElemType e){
if((Q.rear+1)%MAXQSIZE==Q.front) return ERROR;
Q.base[Q.rear]=e;
Q.rear=(Q.rear+1)%MAXQSIZE;
return OK;
}
循环队列出队
Status DeQueue (LinkQueue &Q,QElemType &e){
if(Q.front==Q.rear) return ERROR;
e=Q.base[Q.front];
Q.front=(Q.front+1)%MAXQSIZE;
return OK;
}
=============
链队列
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-0lQCnDfh-1634984533095)(C:\Users\Hasee\AppData\Roaming\Typora\typora-user-images\image-20211023175722297.png)]](https://img-blog.csdnimg.cn/15992925f39e4b15b55e68441bea058f.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBA5q-bX-S4ieaciA==,size_20,color_FFFFFF,t_70,g_se,x_16)
typedef struct QNode{
QElemType data;
struct Qnode *next;
}Qnode, *QueuePtr;
typedef struct {
QueuePtr front;
QueuePtr rear;
}LinkQueue;
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-KOwUEiDR-1634984533096)(C:\Users\Hasee\AppData\Roaming\Typora\typora-user-images\image-20211023175855022.png)]](https://img-blog.csdnimg.cn/cf27b2cdee844fd2887a1d9cd221147f.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBA5q-bX-S4ieaciA==,size_19,color_FFFFFF,t_70,g_se,x_16)
链队列初始化
Status InitQueue (LinkQueue &Q){
Q.front=Q.rear=(QueuePtr) malloc(sizeof(QNode));
if(!Q.front) exit(OVERFLOW);
Q.front->next=NULL;
return OK;
}
销毁链队列
Status DestroyQueue (LinkQueue &Q){
while(Q.front){
Q.rear=Q.front->next;
free(Q.front);
Q.front=Q.rear; }
return OK;
}
判断链队列是否为空
Status QueueEmpty (LinkQueue Q){
return (Q.front==Q.rear);
}
求链队列的队头元素
Status GetHead (LinkQueue Q, QElemType &e){
if(Q.front==Q.rear) return ERROR;
e=Q.front->next->data;
return OK;
}
链队列入队

Status EnQueue(LinkQueue &Q,QElemType e){
p=(QueuePtr)malloc(sizeof(QNode));
if(!p) exit(OVERFLOW);
p->data=e; p->next=NULL;
Q.rear->next=p;
Q.rear=p;
return OK;
}
链队列出队
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-VfdgKtwj-1634984533097)(C:\Users\Hasee\AppData\Roaming\Typora\typora-user-images\image-20211023181201331.png)]](https://img-blog.csdnimg.cn/87e6c7009c704d04a7321fc8acbca7b9.png)
Status DeQueue (LinkQueue &Q,QElemType &e){
if(Q.front==Q.rear) return ERROR;
p=Q.front->next;
e=p->data;
Q.front->next=p->next;
if(Q.rear==p) Q.rear=Q.front;
delete p;
return OK;
}
3.6 案例分析与实现
案例3.1 :数制的转换
【算法步骤】 ① 初始化一个空栈S。 ② 当十进制数N非零时,循环执行以下操作:
- 把N与8求余得到的八进制数压入栈S;
- N更新为N与8的商。
③ 当栈S非空时,循环执行以下操作:
【算法描述】
void conversion(int N)
{//对于任意一个非负十进制数,打印输出与其等值的八进制数
InitStack(S); //初始化空栈S
while(N) //当N非零时,循环
{
Push(S,N%8); //把N与8求余得到的八进制数压入栈S
N=N/8; //N更新为N与8的商
}
while(!StackEmpty(S))//当栈S非空时,循环
{
Pop(S,e); //弹出栈顶元素e
cout<<e; //输出e
}
}
案例3.2 :括号的匹配

|