数据结构与算法分析---栈、队列、表

- 栈

  • 1.1 定义

栈是限制插入和删除只能在一个位置上进行的表,允许操作的一段成为栈顶(top)另一端则称之为栈底,对栈的基本操作有进栈(push)出栈(pop),遵循后进先出(LIFO)原则。

  • 1.2 实现

由于栈是一个受限的表,因此任何方法都能实现栈,一:使用指针,二:使用数组。
指针实现
使用单链表,通过在表前端插入来实现push,通过删除表前端来实现pop。
栈结构体

  • struct Node
    {
    ElementType Element;
    ProToNode Next;
    }*

创建一个空栈
Stack CreateStack(void){
Stack s;
s = malloc (sizeof(struct Node));
if(s ==NULL)
fatalerror(“out of space.”);
s->next == null;
makeempty(s);
return s;
}
void makeempty(Stacks){
if (s == null)
error(“must use createstack first”);
else while(!IsEmpty(s))
pop(s);
}
元素进栈
void push(elementtpye x,stack s){
ptrtonode tmpcell;
tmpcell = malloc (sizeof(struct node)):
if(tmpcell == null )
fatalerror(“out of space.”);
else{
tmpcell->element = x;
tmpcell->next = s->next;
s->next = tmpcell;
}
}
栈的数组实现
避免了因为指针,但这种策略唯一的潜在危险是我们需要提前声明数组的大小。

  • 1.3 应用
    1、平衡符号
    编译器检查程序语法错误,常常由于一个符号的缺失,造成多行代码报错,于是可以使用一个程序来检验是否每个符号都成对出现。
    思想如下:
    创建一个空栈,读入字文件尾符直至文件尾。如果字符是一个开放字符,,入栈。如果字符是一个封闭的字符,则当栈空时报错,否则将栈元素弹出。如果弹出的符号不是对应的开放符号,则报错。在文件尾,非空时报错。
    2、后缀表达式

    3、中缀转后缀

队列

队列也是表,在表的前端(front)进行删除操作,而在表的后端(rear)进行插入操作,和栈一样,队列是一种操作受限制的线性表。进行插入操作的端称为队尾,进行删除操作的端称为队头
在这里插入图片描述
顺序队列
建立顺序队列结构必须为其静态分配或动态申请一片连续的存储空间,并设置两个指针进行管理。一个是队头指针front,它指向队头元素;另一个是队尾指针rear,它指向下一个入队元素的存储位置,如图所示
在这里插入图片描述

每次在队尾插入一个元素是,rear增1;每次在队头删除一个元素时,front增1。随着插入和删除操作的进行,队列元素的个数不断变化,队列所占的存储空间也在为队列结构所分配的连续空间中移动。当front=rear时,队列中没有任何元素,称为空队列。当rear增加到指向分配的连续空间之外时,队列无法再插入新元素,但这时往往还有大量可用空间未被占用,这些空间是已经出队的队列元素曾经占用过得存储单元。
顺序队列中的溢出现象:
(1) "下溢"现象:当队列为空时,做出队运算产生的溢出现象。“下溢”是正常现象,常用作程序控制转移的条件。
(2)"真上溢"现象:当队列满时,做进栈运算产生空间溢出的现象。“真上溢”是一种出错状态,应设法避免。
(3)"假上溢"现象:由于入队和出队操作中,头尾指针只增加不减小,致使被删元素的空间永远无法重新利用。当队列中实际的元素个数远远小于向量空间的规模时,也可能由于尾指针已超越向量空间的上界而不能做入队操作。该现象称为"假上溢"现象。
循环队列
在实际使用队列时,为了使队列空间能重复使用,往往对队列的使用方法稍加改进:无论插入或删除,一旦rear指针增1或front指针增1 时超出了所分配的队列空间,就让它指向这片连续空间的起始位置。自己真从MaxSize-1增1变到0,可用取余运算rear%MaxSize和front%MaxSize来实现。这实际上是把队列空间想象成一个环形空间,环形空间中的存储单元循环使用,用这种方法管理的队列也就称为循环队列。除了一些简单应用之外,真正实用的队列是循环队列。 [2]
在循环队列中,当队列为空时,有front=rear,而当所有队列空间全占满时,也有front=rear。为了区别这两种情况,规定循环队列最多只能有MaxSize-1个队列元素,当循环队列中只剩下一个空存储单元时,队列就已经满了。因此,队列判空的条件时front=rear,而队列判满的条件时front=(rear+1)%MaxSize。队空和队满的情况如图:在这里插入图片描述
当实现循环队列时,一定要警惕第一:检查队列是否为空
第二:某些程序设计人员使用不同的方法来表示队列的对头和队尾,例如有些人并不用一个单元来表示队列的大小,他们用的是基准情形,即当队列为空是,rear = front-1,队列的大小是通过rear 和front 计算出来的
检测是否为空栈
int isempty(queue Q){
return Q->size ==0
}
队列的应用
打印机
接线员

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值