《数据结构教学课件》cha课件.ppt
《《数据结构教学课件》cha课件.ppt》由会员分享,可在线阅读,更多相关《《数据结构教学课件》cha课件.ppt(96页珍藏版)》请在三一办公上搜索。
1、第三章 栈和队列,通常称,栈和队列是限定插入和删除只能在表的“端点”进行的线性表。,线性表 栈 队列Insert(L,i,x)Insert(S,n+1,x)Insert(Q,n+1,x)1in+1 Delete(L,i)Delete(S,n)Delete(Q,1)1in,栈和队列是两种常用的数据类型,3.1 栈的类型定义,3.2 栈类型的实现,3.3 栈的应用举例,3.4 队列的类型定义,3.5 队列类型的实现,3.1 栈的类型定义,ADT Stack 数据对象:D ai|ai ElemSet,i=1,2,.,n,n0 数据关系:R1|ai-1,aiD,i=2,.,n 约定an 端为栈顶,a1
2、 端为栈底。,基本操作:,ADT Stack,栈的特点,栈:限定只能在表的一端进行插入和删除的特殊的线性表 设栈s=(a1,a2,.,ai,.,an),其中a1是栈底元素,an是栈顶元素。,设数组S是一个顺序栈,栈的最大容量stacksize=4,初始状态top=0,栈中的运算:1.设置空栈;2.插入一个新的栈顶元素 3.删除栈顶元素;4.读取栈顶元素。,栈空,10进栈,30出栈,栈满,栈的定义(顺序表),#define STACK_INI_SIZE 100/存储空间初始分配量#define STACKINCREMENT 10/存储空间分配增量typedef structSElemType*b
3、ase;SElemType*top;int stacksize;SqStack;,InitStack(&S),DestroyStack(&S),ClearStack(&S),StackEmpty(s),StackLength(S),GetTop(S,&e),Push(&S,e),Pop(&S,&e),StackTravers(S,visit(),InitStack(&S)操作结果:构造一个空栈 S。,DestroyStack(&S),初始条件:栈 S 已存在。操作结果:栈 S 被销毁。,StackEmpty(S)初始条件:栈 S 已存在。操作结果:若栈 S 为空栈,则返回 TRUE,否则 FA
4、LE。,StackLength(S)初始条件:栈 S 已存在。操作结果:返回 S 的元素个数,即栈的长度。,GetTop(S,&e)初始条件:栈 S 已存在且非空。操作结果:用 e 返回 S 的栈顶元素。,a1,a2,an,ClearStack(&S)初始条件:栈 S 已存在。操作结果:将 S 清为空栈。,Push(&S,e)初始条件:栈 S 已存在。操作结果:插入元素 e 为新的栈顶元素。,a1,a2,an,e,Pop(&S,&e)初始条件:栈 S 已存在且非空。操作结果:删除 S 的栈顶元素,并用 e 返回其值。,a1,a2,an,an-1,3.2栈类型的实现,顺序栈,链栈,/-栈的顺序存
5、储表示-#define STACK_INIT_SIZE 100#define STACKINCREMENT 10 typedef struct SElemType*base;SElemType*top;int stacksize;SqStack;,类似于线性表的顺序映象实现,指向表尾的指针可以作为栈顶指针。,Status InitStack(SqStack,Status Push(SqStack,Status Pop(SqStack 注意:判定栈空的条件S.top=S.base,Status GetTop(SqStack,Status StackEmpty(SqStack S)if(S.top
6、=S.base)return 1;else return 0;,栈顶指针,链栈,a1,an,注意:链栈中指针的方向,an-1,3.3 栈的应用举例,例一、数制转换,例二、括号匹配的检验,例三、行编辑程序问题,例四、迷宫求解,例五、表达式求值,例六、实现递归,例一、数制转换 算法基于原理:N=(N div d)d+N mod d,例如:(1348)10=(2504)8,其运算过程如下:N N div 8 N mod 81348 168 4 168 21 0 21 2 5 2 0 2,计算顺序,输出顺序,void conversion()InitStack(S);scanf(%d,/convers
7、ion,例二、括号匹配的检验假设在表达式中()或()等为正确的格式,()或()或())均为不正确的格式。,则 检验括号是否匹配的方法可用“期待的急迫程度”这个概念来描述。,分析可能出现的不匹配的情况:,到来的右括弧并非是所“期待”的;,例如:考虑下列括号序列:()1 2 3 4 5 6 7 8,到来的是“不速之客”;,直到结束,也没有到来所“期待”的括弧。,算法的设计思想:,1)凡出现左括弧,则进栈;,2)凡出现右括弧,首先检查栈是否空 若栈空,则表明该“右括弧”多余,否则和栈顶元素比较,若相匹配,则“左括弧出栈”,否则表明不匹配。,3)表达式检验结束时,若栈空,则表明表达式中匹配正确,否则表
8、明“左括弧”有余。,Status matching(string.,例三、行编辑程序问题,如何实现?,“每接受一个字符即存入存储器”?,并不恰当!,设立一个输入缓冲区,用以接受用户输入的一行字符,然后逐行存入用户数据区,并假设“#”为退格符,“”为退行符。,在用户输入一行的过程中,允许 用户输入出差错,并在发现有误时 可以及时更正。,合理的作法是:,假设从终端接受了这样两行字符:whli#ilr#e(s#*s)outchaputchar(*s=#+);,则实际有效的是下列两行:while(*s)putchar(*s+);,while(ch!=EOF/从终端接收下一个字符,ClearStack(
9、S);/重置S为空栈if(ch!=EOF)ch=getchar();,while(ch!=EOF)/EOF为全文结束符,将从栈底到栈顶的字符传送至调用过程的数据区;,例四、迷宫求解,通常用的是“穷举求解”的方法,求迷宫路径算法的基本思想是:,若当前位置“可通”,则纳入路径,继续前进;,若当前位置“不可通”,则后退,换方向继续探索;,若四周“均无通路”,则将当前位置从路径中删除出去。,设定当前位置的初值为入口位置;do 若当前位置可通,则将当前位置插入栈顶;若该位置是出口位置,则算法结束;否则切换当前位置的东邻方块为 新的当前位置;否则 while(栈不空);,求迷宫中一条从入口到出口的路径的算
10、法:,若栈不空且栈顶位置尚有其他方向未被探索,则设定新的当前位置为:沿顺时针方向旋转 找到的栈顶位置的下一相邻块;,若栈不空但栈顶位置的四周均不可通,则删去栈顶位置;/从路径中删去该通道块 若栈不空,则重新测试新的栈顶位置,直至找到一个可通的相邻块或出栈至栈空;,若栈空,则表明迷宫没有通路。,限于二元运算符的表达式定义:表达式:=(操作数)+(运算符)+(操作数)操作数:=简单变量|表达式 简单变量:=标识符|无符号整数,例五、表达式求值,表达式的三种标识方法:,设 Exp=S1+OP+S2,则称 OP+S1+S2 为前缀表示法,S1+OP+S2 为中缀表示法,S1+S2+OP 为后缀表示法,
11、一个中缀式到其他式子的转换方法这里给出一个中缀表达式a+b*c-(d+e)第一步:按照运算符的优先级对所有的运算单位加括号式子变成:(a+(b*c)-(d+e)第二步:转换前缀与后缀表达式前缀:把运算符号移动到对应的括号前面则变成拉:-(+(a*(bc)+(de)把括号去掉:-+a*bc+de前缀式子出现后缀:把运算符号移动到对应的括号后面则变成拉:(a(bc)*)+(de)+)-把括号去掉:abc*+de+-后缀式子出现前缀式,后缀式是不需要用括号来进行优先级的确定的。,例如:Exp=a b+(c d/e)f前缀式:+a b c/d e f中缀式:a b+c d/e f后缀式:a b c d
12、 e/f+,结论:,1)操作数之间的相对次序不变;,2)运算符的相对次序不同;,3)中缀式丢失了括弧信息,致使运算的次序不确定;,4)前缀式的运算规则为:连续出现的两个操作数和在它们之前且紧靠它们的运算符构成一个最小表达式;,5)后缀式的运算规则为:运算符在式中出现的顺序恰为 表达式的运算顺序;每个运算符和在它之前出现 且 紧靠它的两个操作数构成一个最小 表达式。,如何从后缀式求值?,先找运算符,再找操作数,例如:a b c d e/f+,ab,d/e,c-d/e,(c-d/e)f,如何从原表达式求得后缀式?,每个运算符的运算次序要由它之后的一个运算符来定,在后缀式中,优先数高的运算符领先于优
13、先数低的运算符。,分析“原表达式”和“后缀式”中的运算符:原表达式:a+b c d/e f 后缀式:a b c+d e/f,从原表达式求得后缀式的规律为:,1)设立操作数栈;,2)设表达式的结束符为“#”,予设运算符栈的栈底为“#”;,3)若当前字符是操作数,则直接发送给后缀式。,4)若当前运算符的优先数高于栈顶运算符,则进栈;,5)否则,退出栈顶运算符发送给后缀式;,6)“(”对它之前后的运算符起隔离作用,“)”可视为自相应左括弧开始的表达式的结束符。,从原表达式求得后缀式的规律为:,void transform(char suffix,char exp)InitStack(S);Push(
14、S,#);Scanf(EXP,ch);/p=exp;ch=*p;while(!StackEmpty(S)if(!IN(ch,OP)Pass(Suffix,ch);else if(ch!=#)p+;ch=*p;else Pop(S,ch);Pass(Suffix,ch);/while/CrtExptree,switch(ch)case(:Push(S,ch);break;case):Pop(S,c);while(c!=()Pass(Suffix,c);Pop(S,c)break;defult:while(Gettop(S,c)/switch,将中缀表达式转换为前缀表达式:(1)初始化两个栈:运算
15、符栈S1和储存中间结果的栈S2;(2)从右至左扫描中缀表达式;(3)遇到操作数时,将其压入S2;(4)遇到运算符时,比较其与S1栈顶运算符的优先级:(4-1)如果S1为空,或栈顶运算符为右括号“)”,则直接将此运算符入栈;(4-2)否则,若优先级比栈顶运算符的较高或相等,也将运算符压入S1;(4-3)否则,将S1栈顶的运算符弹出并压入到S2中,再次转到(4-1)与S1中新的栈顶运算符相比较;(5)遇到括号时:(5-1)如果是右括号“)”,则直接压入S1;(5-2)如果是左括号“(”,则依次弹出S1栈顶的运算符,并压入S2,直到遇到右括号为止,此时将这一对括号丢弃;(6)重复步骤(2)至(5),
16、直到表达式的最左边;(7)将S1中剩余的运算符依次弹出并压入S2;(8)依次弹出S2中的元素并输出,结果即为中缀表达式对应的前缀表达式。,将中缀表达式转换为后缀表达式:与转换为前缀表达式相似,遵循以下步骤:(1)初始化两个栈:运算符栈S1和储存中间结果的栈S2;(2)从左至右扫描中缀表达式;(3)遇到操作数时,将其压入S2;(4)遇到运算符时,比较其与S1栈顶运算符的优先级:(4-1)如果S1为空,或栈顶运算符为左括号“(”,则直接将此运算符入栈;(4-2)否则,若优先级比栈顶运算符的高,也将运算符压入S1(注意转换为前缀表达式时是优先级较高或相同,而这里则不包括相同的情况);(4-3)否则,
- 配套讲稿:
如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
- 特殊限制:
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。
- 关 键 词:
- 数据结构教学课件 数据结构 教学 课件 cha

链接地址:https://www.31ppt.com/p-5030410.html