数据结构第三章课件_第1页
数据结构第三章课件_第2页
数据结构第三章课件_第3页
数据结构第三章课件_第4页
数据结构第三章课件_第5页
已阅读5页,还剩168页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

1第三章栈和队列2栈的定义、表示及实现;栈的应用(表达式求值、递归);队列的定义、表示及实现栈和队列的特点;在两种存储结构上栈的基本操作的实现;循环队列和链队列的基本运算;递归算法执行过程中栈状态的变化过程循环队列和链队列的基本运算。

教学内容:※教学重点:

※教学难点:

3

栈和队列是限定插入和删除只能在表的“端点”进行的线性表。

线性表栈队列Insert(L,

i,x)Insert(S,n+1,x)Insert(Q,n+1,x)

1≤i≤n+1Delete(L,i)Delete(S,n)Delete(Q,1)

1≤i≤n栈和队列是两种常用的数据类型43.1栈的类型定义3.3栈的应用举例3.2栈类型的实现3.4队列的类型定义3.5队列类型的实现5一、栈(stack)的定义

限定仅在表尾进行插入或删除操作的线性表。其中允许进行插入和删除的一端(表尾)称为栈顶;另一端(表头)称为栈底。当表中没有元素时,称为空栈。3.1栈的类型定义6

假设栈

S=(a1,a2,…,an)栈底元素栈顶元素进栈次序:

a1,a2,…,an退栈次序:an,an-1,…,a2,a1a1a2an栈顶栈底进栈退栈栈是按照“后进先出”原则处理数据元素的,栈也称为“后进先出”表。简称LIFO表。栈顶栈顶7二、栈的抽象数据类型的类型定义

ADTStack

{

数据对象:

D={ai|ai∈ElemSet,i=1,2,...,n,n≥0}

数据关系:

R1={<ai-1,ai>|ai-1,ai∈D,i=2,...,n}

约定an

端为栈顶,a1端为栈底。

基本操作:

}ADTStack8InitStack(&S)DestroyStack(&S)ClearStack(&S)StackEmpty(s)StackLength(S)GetTop(S,&e)Push(&S,e)Pop(&S,&e)StackTravers(S,visit())9

InitStack(&S)初始化操作

操作结果:

DestroyStack(&S)

初始条件:

操作结果:栈S已存在。

栈S被销毁。构造一个空栈S。10

StackEmpty(S)判定S是否为空栈

初始条件:

操作结果:

StackLength(S)求栈的长度初始条件:

操作结果:栈S已存在。返回S的元素个数,即栈的长度。

栈S已存在。

若栈S为空栈,则返回TRUE,

否则FALSE。11GetTop(S,&e)取栈顶元素

初始条件:

操作结果:a1a2an……栈S已存在且非空。

用e返回S的栈顶元素。12ClearStack(&S)栈置空操作

初始条件:

操作结果:栈S已存在。

将S清为空栈。13Push(&S,e)入栈操作

初始条件:

操作结果:a1a2ane……栈S已存在。

插入元素e为新的栈顶元素。top14Pop(&S,&e)出栈操作

初始条件:

操作结果:a1a2anan-1

……栈S已存在且非空。

删除

S的栈顶元素,并用e返回其值。top15

3-1

有三个元素的进栈顺序为1、2、3。举出此三个元素可能的出栈序列,并写出相应的进栈和出栈操作序列(假设以s和x分别表示进栈和出栈操作)。

出栈序列

1、2、31、3、22、1、32、3、13、2、1操作序列sxsxsxsxssxxssxxsxssxsxxsssxxx16利用一组地址连续的存贮单元(数组)依次存放从栈底到栈顶的若干数据元素,数组的上界maxsize表示栈的最大容量;附设栈顶指针Top来指示栈顶元素在数组中的位置。一个栈的栈底位置是固定的,栈顶位置随着进栈和出栈操作而变化。一、栈的顺序存储结构3.2栈类型的实现17ABCDTop=5Top=-1(空栈,再出栈产生“下溢”)(栈满,再入栈产生“上溢”)ABCABTop=2Top=1Top=-1(空栈)(A.B.C入栈后栈的状态)(C出栈后的)EF18C语言描述:类似于线性表的顺序存贮结构

#defineMAXSIZEn

/*n为栈中数据元素个数的最大可能值*/

typedefstruct{}sqstack;

栈的顺序存储结构方法一elemtypestack[MAXSIZE];inttop;

19voidInitstack(sqstack&s){

}初始化栈算法s.top=-1;//设置栈顶指针top,表示栈空20statuspush(sqstack&s,elemtypex){if(s.top>=MAXSIZE-1)returnERROR;//栈满,上溢

else{s.top++;

s.stack[top]=x;returnOK;}}//push进栈算法21出栈算法status

pop(sqstack&s,elemtype&e)

{if(s.top<0)returnERROR;//栈空,下溢

else{e=s.stack[top];s.top--;returnOK;}}//pop22取栈顶元素算法status

gettop(sqstacks,elemtype&e)

{if(s.top<0)returnNULL;//栈空,返回空值

else{e=s.stack[top];returnOK;}}//gettop23判栈空算法statusempty(sqstacks)

{if(s.

top<0)returnTRUE;//栈空,返回TRUEelse{returnFALSE;}}//empty24

在实现上述这些操作时,可以不把栈S定义为结构体类型(sqstacks)

而是定义为指向结构体的指针类型:这样在编程时,只需将

s.top改为s

top

s.stack[]改为s

stack[]即可。Sqstack*s;

25#defineSTACK_INIT_SIZE100;#defineSTACKINCREMENT10;

typedefstruct{

SElemType

*base;

SElemType

*top;//指向栈顶的下一个位置

int

stacksize;

}

SqStack;

类似于线性表的顺序映象实现,指向表尾的指针可以作为栈顶指针。栈顶指针指向最后一个元素的下一个位置。栈的顺序存储结构方法二26

StatusInitStack(SqStack&S){//构造一个空栈SS.base=(ElemType*)malloc(STACK_INIT_SIZE*sizeof(ElemType));

if(!S.base)exit(OVERFLOW);//存储分配失败

}S.top=S.base;S.stacksize=STACK_INIT_SIZE;

returnOK;27

StatusPush(SqStack&S,SElemTypee){if(S.top-S.base>=S.stacksize){//栈满,追加存储空间

S.base=(ElemType*)realloc(S.base,(S.stacksize+STACKINCREMENT)*

sizeof(ElemType));

if(!S.base)exit(OVERFLOW);//存储分配失败

S.top=S.base+S.stacksize;S.stacksize+=STACKINCREMENT;

}

*S.top++=e;//先赋值,top后加1

returnOK}入栈28StatusPop(SqStack&S,SElemType&e){

//若栈不空,则删除S的栈顶元素,

//用e返回其值,并返回OK;

//否则返回ERROR

if

(S.top==S.base)

returnERROR;

e=*--S.top;//先top减1,后取值给ereturnOK;}出栈29顺序栈判满、判空条件

顺序栈空时:注意:

非空栈中的栈顶指针始终在栈顶元素的下一个位置上。顺序栈满时:S.top-S.base>=S.stacksizeS.top=S.base30

如果在一个程序中需要使用多个栈,为了充分利用各个栈的空间,不产生栈空间过大造成系统空间紧张的问题,又要保证各个栈都能足够大以保存入栈的元素,不产生“上溢”现象,最好能采用多个栈共享空间的方法,即给多个栈分配一个足够大的数组空间,利用栈的动态特性,使其存储空间互相补充。为说明简单,假定在程序中同时使用两个栈。给它们分配一个长度为m的数组空间,这时可将这两个栈的栈底设在数组的两端,让它们的栈顶向数组中间增长。这样当一个栈中元素较多时,就可以越过数组的中点,延伸到另一个栈的部分空间中去,当然前提是另一个栈中当前的元素不多。只有当两个栈的栈顶相迂时,才会发生“上溢”。显然,这比两个各有m/2存储空间的栈产生“上溢”的概率要小。31栈的顺序存储结构方法三(多个栈共享空间)0m-1Top1(栈1顶)Top2(栈2顶)这种栈的数据结构可定义为:typedefstruct{elemtypestack[m];

inttop1,top2;

}dustack;

栈1底栈2底32top2=mtop1=-1栈1栈1栈2栈2(a)栈1和栈2均为空栈时(b)栈1和栈2均有若干元素进栈时(c)栈满时top1top2top2top1两个栈共享一段存贮空间33

例子:有两个栈S1和S2共享存储空间c[1,m0],其中一个栈底设在c[1]处,另一个栈底设在c[m0]处,分别编写S1和S2的进栈push(i,x)、退栈pop(i)和设置栈空setnull(i)的函数,其中,i=1,2。注意:仅当整个空间c[1,m0]占满时才产生上溢。34a1a2……an……bm……b2b1C的元素序号

12……n……m0-1m0

栈1底

栈1顶栈2顶栈2底

top1top2

该共享栈的结构如图所示,两栈的最多元素个数为m0,top1是栈1的栈顶指针,top2是栈2的栈顶指针,当top2=top1+1时出现上溢;当top1=0时栈1出现下溢,当top2=m0+1时栈2出现下溢。35根据上述原理得到入栈函数如下:voidpush(inti,intx){}if(top1==top2-1)printf(“上溢!\n”);elseif(i==1);//对第一个栈进行入栈操作

{top1++;c[top1]=x;}else//对第二个栈进行入栈操作

{top2--;c[top2]=x;}36voidpop(inti){if(i==1);//对第一个栈进行出栈操作

if(top1==0)printf(“栈1下溢!\n”);else{x=c[top1];top1--;}else//对第二个栈进行出栈操作

if(top2==m0+1)printf(“栈2下溢!\n”)

else{x=c[top2];top2++;}}出栈函数如下:37voidsetnull(inti){if(i==1);top1=0;elsetop2=m0+1;}判栈空函数如下:38

当最大需要容量事先不能估计时,采用链式存贮结构是有效的方法,链栈的操作只能在栈顶处进行。二、栈的链式存贮结构a5a4a3a2a1^栈底栈顶topdatanexttypedefstructsnode{elemtypedata;

structsnode*next;}*linkstack;39statuspush(linkstacktop,elemtypex){}//push链式进栈操作t=(linkstack)malloc(sizeof(snode));if(t==NULL)returnERROR;//内存无可用空间,栈上溢

else{t->data=x;t->next=top;top=t;returnOK;}40status

pop(linkstacktop,elemtype&e){}//pop链式出栈操作qif(top==NULL)//栈空,栈溢出(下溢)

returnNULL;else{p=top;top=top->next;e=p->data;free(p);returnOK;}41

对于链栈,不会产生单个栈满而其余栈空的情形,只有当系统空间全部用完,malloc过程无法实现时才会发生上溢,因此多个链栈共享空间也就是自然的事了。可见,对栈这样一种元素多变的数据结构来说,链式存储结构似乎更适宜些。此外初始化栈、栈判空操作也很容易实现。423.3栈的应用举例例一、数制转换例二、括号匹配的检验例三、行编辑程序问题例四、迷宫求解例五、表达式求值例六、实现递归43

例一、数制转换

算法基于原理:

N=(Ndivd)×d+Nmodd

44例如:

(1348)10=(2504)8,其运算过程如下:

NNdiv8Nmod8

210

2125

202计算顺序输出顺序45void

conversion(){}

//conversionInitStack(S);

scanf("%d",N);while(N){//N不等于0时循环

Push(S,N%8);N=N/8;

}while(!StackEmpty(S)){

Pop(S,e);

printf("%d",e);

}46#definestacksize100;typedefstruct{intbase[stacksize];inttop;}stack;

数制转换的完整程序47intpush(stack*s,inte){

if(s->top>=stacksize)return0;s->base[s->top]=e;s->top++;return1;}intpop(stack*s,int*e){

if(s->top==0)return0;*e=s->base[--s->top];return1;}//top指向栈顶元素的下一个位置48main(){stacks1;intm,e,n;s1.top=0;m=1348;n=8;while(m){push(&s1,m%n);m=m/n;}while(s1.top!=0){pop(&s1,&e);printf("%d",e);}}49例二、括号匹配的检验假设在表达式中,

([]())或[([][])]则检验括号是否匹配的方法可用“期待的急迫程度”这个概念来描述。

等为正确的格式,[(])或([())或(()])

均为不正确的格式。50分析可能出现的不匹配的情况:到来的右括弧并非是所“期待”的;例如:考虑下列括号序列:

[([][])]12345678到来的是“不速之客”;直到结束,也没有到来所“期待”的括弧。51算法的设计思想:1)凡出现左括弧,则2)凡出现右括弧,3)表达式检验结束时,

若栈空,则表明表达式中匹配正确,

否则表明“左括弧”有余。进栈;“右括弧”多余,首先检查栈是否空若栈空,则表明该

否则和栈顶元素比较,

若相匹配,则“左括弧出栈”

否则表明不匹配。52Statusmatching(string&exp){//只有圆括号,调用基本操作}//matchingintstate=1;i=1;

while(i<=Length(exp)&&state){

switch(exp[i])

{

case

“(”:{Push(S,exp[i]);i++;break;}

case”)”:{

if(NOTstackEmpty(S)&&GetTop(S)=“(“){Pop(S,e);i++;}

elsestate=0;break;}//表示不匹配

default:{

state=0;break;}}if(StackEmpty(S)&&state)return

OK;

else

returnERROR53例三、行编辑程序问题

一个简单的行编辑程序的功能是:接收用户从终端输入的程序或数据,并存入用户的数据区。如何实现?“每接受一个字符即存入存储器”?

由于用户在终端上进行输入时,不能保证不出差错,因此,若在编辑程序中,“每接收一个字符即存入用户数据区”的做法显然不是最恰当的。54

实现:设立一个输入缓冲区,用以接受用户输入的一行字符,然后逐行存入用户数据区,并假设“#”为退格符(前一个字符无效),“@”为退行符。合理的作法是:

在用户输入一行的过程中,允许用户输入出差错,并在发现有误时可以及时更正。55假设从终端接受了这样两行字符:

whli##ilr#e(s#*s)

outcha@putchar(*s=#++);则实际有效的是下列两行:

while(*s)

putchar(*s++);56

为此,可设这个输入缓冲区为一个栈结构,每当从终端接收了一个字符之后先作如下判别:

如果它既不是退格符也不是退行符,则将该字符压入栈顶;如果是一个退格符,则从栈顶删去一个字符;如果是一个退行符,则将字符栈清为空栈。可用下述算法来描述:57voidLineEdit(){InitStack(S);//构造空栈

ch=getchar();//从终端接收第一个字符

while(ch!=EOF)//EOF是全文结束符

{while(ch!=EOF&&ch!=‘\n’){switch(ch){case‘#’:Pop(S,ch);break;case‘@’:ClearStack(S);break;

//重置S为空栈

default:Push(S,ch);break;}ch=getchar();

//从终端接收下一个字符

}

将从栈底至栈顶的栈内字符传送至调用过程的数据区;

ClearStack(S);//重置S为空栈

if(ch!=EOF)ch=getchar();}DestroyStack(S);//栈S被销毁

}//LineEdit58通常用的是“穷举求解”的方法#############################################Q###########例四、迷宫求解→→↓↓→→↑→←↑←↓$$$$$$$$↓→→↓→→↓↓→→→→←从点的东向开始按顺时针旋转入口59求迷宫路径算法的基本思想是:若当前位置“可通”,则纳入路径,继续前进;若当前位置“不可通”,则后退,换方向继续探索;若四周“均无通路”,则将当前位置从路径中删除出去。迷宫图60设定当前位置的初值为入口位置;

do{若当前位置可通,则{将当前位置插入栈顶;若该位置是出口位置,则算法结束;

否则切换当前位置的东邻方块为

新的当前位置;

否则{

}}while(栈不空);求迷宫中一条从入口到出口的路径的算法:

A:…

…61A:若栈不空且栈顶位置尚有其他方向未被探索,

{则设定新的当前位置为:沿顺时针方向旋转找到的栈顶位置的下一相邻块;}若栈不空但栈顶位置的四周均不可通,

{则删去栈顶位置;//从路径中删去该通道块

若栈不空,则重新测试新的栈顶位置,直至找到一个可通的相邻块或出栈至栈空;}

若栈空,则表明迷宫没有通路。62

表达式求值是程序设计语言编译中的一个基本问题,它的实现是栈应用的又一个典型例子。这里介绍一种简单直观,广为使用的算法,叫“算符优先法”。它是根据运算优先关系的规定来实现对表达式的编译或解释执行的。例如:要对算术表达式4+2*3-10/5求值。算术四则运算规则:⑴先括号内,后括号外;⑵先乘除,后加减;⑶同级运算从左算到右。“算符优先法”就是根据这个运算优先关系的规定来实现对表达式的编译或解释执行的。例五、表达式求值63表达式操作数(operand):常数、变量或常量标识符运算符(operator):算术运算符、关系运算符、逻辑运算符等界限符(delimiter):(、)、表达式结束符等

为了叙述的简洁,我们仅讨论简单算术表达式的求值问题,这种表达式仅包含加、减、乘、除、括号等四种运算。不难将它推广到更一般的表达式上。64

我们把运算符和界限符统称为算符,它们构成的集合命名为op,在运算的每一步中,任意两个相继出现的运算符θ1和θ2之间的优先关系至多是下面三种关系之一:

θ1<θ2:θ1的优先权低于θ2;

θ1=θ2:θ1的优先权等于θ2;

θ1>θ2:θ1的优先权高于θ2;下表定义了算符之间的这种优先关系。65算符优先关系表

+-*/

()#>><<<>>>><<<>>>>>><>>>>>><>><<<<<=>>>>>><<<<<=+-*/()#θ1θ266

为实现算符优先算法可使用两个工作栈:

OPTR栈:用以寄存运算符;

OPND栈:用以寄存操作数或运算结果;算法基本思想:⑴首先置操作数栈为空栈,表达式起始符“#”为运算符栈的栈底元素;⑵依次读入表达式中的每个字符,若是操作数,则进OPND栈;若是运算符,则和OPTR栈的栈顶运算符比较优先数后作相应操作,直至整个表达式求值完毕(即OPTR栈的栈顶元素和当前读入的字符均为“#”)。67OperandTypeEvaluateExpression()//求算术表达式的值

{InitStack(OPTR);Push(OPTR,’#’);//初始化运算符栈

InitStack(OPND);c=getchar();//初始化操作数栈

while(c!=‘#’||GetTop(OPTR))!=‘#’){if(!In(c,OP)//读入的c不是运算符

{Push((OPND,c);c=getchar();}//操作数进栈

elseswitch(Precede(GetTop(OPTR),c){case‘<’:Push(OPTR,c);c=getchar();break;//栈顶元素优先权低,运算符进栈

case‘=’:Pop(OPTR,x);c=getchar();break;//脱括号并接收下一字符

case‘>’:Pop(OPTR,theta);Pop(OPND,b);Pop(OPND,a);Push(OPND,Operate(a,theta,b));break;//退栈并将运算结果入栈

}//switch}//whilereturnGetTop(OPND);}//EvaluateExpression

68

表达式::=(操作数)+(运算符)+(操作数)

操作数::=简单变量|表达式简单变量::=标识符|无符号整数二元运算符的表达式的三种标识方法69

表达式的三种标识方法:设

Exp=S1+

OP

+S2则称

OP

+S1+S2

为前缀表示法

S1+

OP

+S2

为中缀表示法

S1+S2+

OP

为后缀表示法

70例如:Exp=a

b

+

(c

d/e)

f前缀式:

中缀式:

后缀式:

结论:1)操作数之间的相对次序不变;2)运算符的相对次序不同;3)中缀式丢失了括弧信息,致使运算的次序不确定;

+

ab

c/defa

b

+

c

d/e

fab

cde/

f

+715)后缀式的运算规则为:

每个运算符和在它之前出现

且紧靠它的两个操作数构成一个最小表达式;

运算符在表达式中出现的顺序恰为表达式的运算顺序。4)前缀式的运算规则为:

连续出现的两个操作数和在它们之前且紧靠它们的运算符构成一个最小表达式;72如何从后缀式求值?先找运算符,再找操作数例如:

ab

cde/

f

+a

bd/ec-d/e(c-d/e)

f73如何从原表达式求得后缀式?

每个运算符的运算次序要由它之后的一个运算符来定,在后缀式中,优先数高的运算符领先于优先数低的运算符。分析“原表达式”和“后缀式”中的运算符:原表达式:

a+b

c

d/e

f后缀式:

abc

+de/f

74从原表达式求得后缀式的规律为:1)设立运算符栈;2)设表达式的结束符为“#”,

预设运算符栈的栈底为“#”;3)若当前字符是操作数,则直接发送给后缀式。754)若当前运算符的优先数高于栈顶运算符,则进栈;5)否则,退出栈顶运算符发送给后缀式;

6)“(”对它之前后的运算符起隔离作用,“)”可视为自相应左括弧开始的表达式的结束符。从原表达式求得后缀式的规律为:76voidtransform(charsuffix[],charexp[])//从原表达式求得后缀式

//s是运算符栈,s栈底预设‘#’,OP是运算符集合

{}//CrtExptreeInitStack(S);Push(S,

#

);p=exp;ch=*p;

while(!StackEmpty(S)){if(!IN(ch,OP))//若ch是操作数

Pass(Suffix,ch);else{A:}//若ch是运算符

if(ch!=

#

){p++;ch=*p;}else{Pop(S,ch);Pass(Suffix,ch);}

}//while……77A: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)&&(precede(c,ch)))

{Pass(Suffix,c);Pop(S,c);}

if(ch!=

#

)Push(S,ch);

break;

}//switch若ch的优先级比c低,为真781)设立操作数栈S;2)设表达式的结束符为“#”;3)读入表达式一个字符,若当前字符是操作数,则压入栈中,转4)。求后缀式的值:79若当前字符是运算符optr,从栈中弹出2个数,将运算结果再压入栈,即:x2=POP(S);x1=POP(S);

PUSH(S,(x1optrx2));读下一字符,重复3)和4)直至读到结束符#;

栈顶,x=POP(S)即后缀式相应的表达式的值。80intcal(charsuffix-exp[])//求后缀式表达式的值

//s是运算数栈,OP是运算符集合

{

}//calInitStack(S);i=0;ch=suffix-exp[0];while(ch<>’#’){if(!IN(ch,OP))//若ch是操作数

Push(S,ch);else//若ch是运算符

{x2=POP(S);x1=POP(S);//取出两个操作数

PUSH(S,(x1chx2));}i++;ch=suffix-exp[i];}//while81例六、实现递归将所有的实在参数、返回地址等信息传递给被调用函数保存;为被调用函数的局部变量分配存储区;将控制转移到被调用函数的入口。

当在一个函数的运行期间调用另一个函数时,在运行该被调用函数之前,

需先完成三项任务:82保存被调函数的计算结果;释放被调函数的数据区;依照被调函数保存的返回地址将控制转移到调用函数。从被调用函数返回调用函数之前,应该完成下列三项任务:83多个函数嵌套调用的规则是:此时的内存管理实行“栈式管理”后调用先返回!例如:voidmain(){voida(){voidb(){

………

a();b();

……}//main}//a}//bMain的数据区函数a的数据区函数b的数据区函数a的数据区84

函数之间的信息传递和控制转移必须通过“栈”来实现,系统将整个程序运行时所需要的数据空间安排在一个栈内,每当调用一个函数时,就为它在栈顶分配一个存储区,每当从一个函数退出时,就释放它的存储区。所以当前运行的函数的数据区必在栈顶。一个递归函数的运行过程类似于多个函数的嵌套调用,只是调用函数与被调用函数是同一个函数。注意:递归函数运行的“层次”。主函数是第0层,从主函数调用递归函数为进入第1层。每进入一层递归,就产生一个新的工作记录(包括上一层的返回地址、实在参数、局部量)压入栈顶。每退出一层递归,就从栈顶弹出一个工作记录。85例3-2n阶Hanoi问题。假设有三个分别命名为X、Y和Z的塔座,在塔座X上插有n个直径大小各不相同、依小到大编号为1,2,…,n的圆盘。现要求将X轴上的n个圆盘移至塔座Z上,并仍按同样顺序叠排。12..n-1nXYZ12..nXYZ86圆盘移动时必须遵循下列规则:1)每次只能移动一个圆盘;

2)圆盘可以插在X、Y和Z中的任一塔座上;

3)任何时刻都不能将一个较大的圆盘压在较小的圆盘之上。XYZ87如何实现圆盘的移动操作?当n=1时,从塔座X

Z;XYZ

而如何将n-1个圆盘从一个塔座移至另一个塔座的问题是一个和原问题具有相同特征属性的问题,只是问题的规模小1,因此可以用同样的方法求解。XYZ

当n>1时,利用塔座Y作辅助塔座,若能设法将压在编号为n的圆盘之上的n-1个圆盘从塔座X

Y上,则可先将编号为n的圆盘从塔座X

Z,然后再将塔座Y上的n-1个圆盘移至塔座Z上。88voidhanoi(intn,charx,chary,charz)//

将塔座x上按直径由小到大且至上而下编号为1至n//

的n个圆盘按规则搬到塔座z上,y可用作辅助塔座。

1{2if(n==1)3move(x,1,z);//

将编号为1的圆盘从x移到z4else{5hanoi(n-1,x,z,y);//

将x上编号为1至n-1的

//圆盘移到y,z作辅助塔

6move(x,n,z);//

将编号为n的圆盘从x移到z7hanoi(n-1,y,x,z);//

将y上编号为1至n-1的

//圆盘移到z,x作辅助塔8}9}8903abc返址nxyz52acb51abc71cabvoidhanoi(intn,charx,chary,charz){1if(n==1)2move(x,1,z);3else4

{hanoi(n-1,x,z,y);5move(x,n,z);6hanoi(n-1,y,x,z);7}8}

语句标号90过程的嵌套调用r主程序srrrs子过程1rst子过程2rst子过程3栈的应用91地图四染色问题“四染色”定理是计算机科学中著名的定理之一。使地图中相邻的国家或行政区域不重色,最少可用四种颜色对地图着色。证明此定理的结论,利用栈采用回溯法对地图着色。思想:对每个行政区编号:1-7

对颜色编号;a、b、c、d;从第一号行政区开始逐一染色,每一个区域逐次用四种颜色进行试探,若所取颜色与周围不重,则用栈记下来该区域的色数,否则依次用下一色数进行试探。若出现a-d均与周围发生重色,则需退栈回溯,修改当前栈顶的色数。92(1)(2)(4)(5)(6)(7)(3)R[7][7]12345671234567100001001111101010110101101011011001001100000000012345671223414334231#

紫色

2#黄色3#

红色4#

蓝色地图四染色举例S[K]93Voidmapcolor(intR[][],intn,ints[]){s[1]=1;//1号区域染1色

a=2;J=1;//a为区域号,J为染色号

while(a<=n)

{while((J<=4)&&(a<=n)){k=1;//k表示已经着色的区域号

while((k<a)&&(s[k]

R[a-1][k-1]!=J))k=k+1;

//若不相邻,或若相邻且不重色,对下一个区域判断。

IF(k<a)J=J+1;//相邻且重色

ELSE{s[a]=J;a=a+1;J=1;}//相邻且不重色

}IF(J>4){a=a-1;J=s[a]+1}}}94

设n皇后问题的解为(x1,x2,x3,…,xn),

约束条件为:其中任意两个xi

和xj不能位于棋盘的同行、同列及同对角线。如何表示棋盘中放置的棋子?

由于每行、列及对角线上只能有一个棋子,因而对每列来说,只需记录该列中棋子所在的行号,故用一维数组即可。皇后问题求解95

设四皇后问题的解为(x1,x2,x3,x4),

其中:xi(i=1,2,3,4)Si={1,2,3,4}约束条件为:其中任意两个xi

和xj不能位于棋盘的同行、同列及同对角线。

按回溯法的定义,皇后问题求解过程为:解的初始值为空;首先添加x1=1,之后添加满足条件的x2=3,由于对所有的x3

{1,2,3,4}都不能找到满足约束条件的部分解(x1,x2,x3),

则回溯到部分解(x1),

重新添加满足约束条件的x2=4,依次类推(按行存列号)。按四皇后问题求解举例96••••••••••••••••••••••••••••••••••••••••••••••••••••••••••••••••••••••••97void

queen(inti,intn)

{//进入本函数时,在n×n棋盘前i-1行已放置了互不攻

//击的i-1个棋子。现从第i行起继续为后续棋子选择

//满足约束条件的位置。当求得(i>n)的一个合法布局

//时,输出之。

if(i>n)输出棋盘的当前布局;

elsefor(j=1;j<=n;++j)

{

在第i行第j列放置一个棋子;

if(当前布局合法)queen(i+1,n);

移去第i行第j列的棋子;

}}//trial98#include<stdio.h>#definen8//n为皇后个数,m为摆法计数intm=0,a[n];

//a[i]存放第i个皇后放置的行号,int

ok(inti,intj)

//检查(i,j)上能否放棋子{

j1=j;i1=i;ok1=1;//检查第i行上能否放棋子

while((j1>1)&&ok1){j1--;ok1=a[j1]!=i;}

j1=j;i1=i;//检查对角线上能否放棋子

while((j1>1)&&(i1>1)&&ok1){j1--;i1--;ok1=a[j1]!=i1;}j1=j;i1=i;//检查另一对角线上能否放棋子

while((j1>1)&&(i1<n)&&ok1){j1--;i1++;ok1=a[j1]!=i1;}returnok1}99Voidqueen(intj)

//从第j列开始逐个试探{

if(j>n)

{m++;

printf("m=%d",m);

for(i=1;i<=n;i++)printf("%d",a[i]);

printf(“\n”);

}

elsefor(i=1;i<=n;i++)if(ok(i,j))//检查(i,j)上能否放棋子

{a[j]=i;//在(i,j)上放一个棋子

queen(j+1);}}main(){queen(1);}100队列是只允许在表的一端进行插入,在另一端删除元素的线性表;允许插入的一端称为队尾(rear);允许删除的一端称为队头(front);假设队列为Q=(a1,a2,…,an),则a1是队头元素,an是队尾元素;队列中的元素是按照a1,a2,…,an的顺序进入的,退出队列也只能按照这个次序依次退出;当队列中没有元素时称为空队列;队列是一种“先进先出”的线性表,简称FIFO表。3.4队列的类型定义一、队列的定义101a1

a2

a3…an出队列

入队列队头队尾队列示意图102

ADTQueue{

数据对象:

D={ai|ai∈ElemSet,i=1,2,...,n,n≥0}

数据关系:

R1={<ai-1,ai>|ai-1,ai∈D,i=2,...,n}

约定其中a1

端为队列头,an

端为队列尾基本操作:}ADTQueue二、队列的抽象数据类型的类型定义103队列的基本操作:

InitQueue(&Q)DestroyQueue(&Q)QueueEmpty(Q)QueueLength(Q)GetHead(Q,&e)ClearQueue(&Q)DeQueue(&Q,&e)EnQueue(&Q,e)QueueTravers(Q,visit())104InitQueue(&Q)

操作结果:构造一个空队列Q。

DestroyQueue(&Q)

初始条件:队列Q已存在。

操作结果:队列Q被销毁,

不再存在。

105

QueueEmpty(Q)

初始条件:队列Q已存在。

操作结果:若Q为空队列,则返回

TRUE,否则返回FALSE。106

QueueLength(Q)

初始条件:队列Q已存在。

操作结果:返回Q的元素个数,

即队列的长度。107

GetHead(Q,&e)

初始条件:Q为非空队列。

操作结果:用e返回Q的队头元素。a1a2an……108

ClearQueue(&Q)

初始条件:队列Q已存在。

操作结果:将Q清为空队列。109

EnQueue(&Q,e)

初始条件:队列Q已存在。

操作结果:插入元素e为Q的新的

队尾元素。a1a2ane……110

DeQueue(&Q,&e)

初始条件:Q为非空队列。

操作结果:删除Q的队头元素,

并用e返回其值。a1a2an……

111QueueTravers(Q,visit())

初始条件:Q为非空队列。

操作结果:从队头到队尾,依次对Q

的每个数据元素调用函数

visit()。一旦Visit()失败,则操作失败。1123.5队列类型的实现链队列——链式映象循环队列——顺序映象1131、队列的顺序存贮结构

队列的顺序存储结构称为顺序队列,采用一维数组来存储队列中的每一元素,数组的上界表示队列的最大容量。另设两个指针front

和rear

,分别指示队头元素和队尾元素在队列中的位置。约定:

队头指针front总是指向队头元素在队列中的当前位置;队尾指针rear总是指示队尾元素的下一个位置;114C语言描述:

#defineMAXQSIZEntypedefstruct{elemtypequeue[MAXQSIZE];//静态分配

intfront,rear;}sequeuetp;

ABCDEfrontrear(1)一般的队列115C语言描述:

#defineMAXQSIZEntypedefstruct{elemtype*queue;//动态分配

intfront,rear;}sequeuetp;116例:顺序队列Q的C语言描述是:

#defineMAXQSIZE8typedefstruct{charqueue[MAXQSIZE];intfront,rear;}sequeuetp;sequeuetpQ;

117(a)队列的初始状态(空队列):01234567Q.frontQ.rear

状态特征:Q.front=Q.rear=0118初始化算法voidinitqueue(sequeuetpQ){Q.front=0;Q.rear=0;//设置队头指针和队尾指针均指向

//数组下界

}119

(b)

元素入队列后,队列的状态:

Q.frontQ.rear

入列操作:

Q.queue[Q.rear]=‘A’;Q.rear++;Q.queue[Q.rear]=‘B’;Q.rear++;Q.queue[Q.rear]=‘C’;Q.rear++;Q.queue[Q.rear]=‘D’;Q.rear++;Q.queue[Q.rear]=‘E’;Q.rear++;ABCDE120

(c)

元素A,B,C出列后的状态:Q.frontQ.rear

出列操作:

ch=Q.queue[Q.front];Q.front++;ch=Q.queue[Q.front];Q.front++;ch=Q.queue[Q.fr

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论