




版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领
文档简介
第5章递归
3.3递归算法的设计5.1什么是递归5.2递归调用的实现原理本章小结15.1什么是递归5.1.1递归的定义
在定义一个过程或函数时出现调用本过程或本函数的成分,称之为递归。若调用自身,称之为直接递归。若过程或函数p调用过程或函数q,而q又调用p,称之为间接递归。如果一个递归过程或递归函数中递归调用语句是最后一条执行语句,则称这种递归调用为尾递归。2例5.1以下是求n!(n为正整数)的递归函数。
intfun(intn){if(n==1) /*语句1*/return1; /*语句2*/else /*语句3*/return(fun(n-1)*n); /*语句4*/}
在该函数fun(n)求解过程中,直接调用fun(n-1)(语句4)自身,所以它是一个直接递归函数。又由于递归调用是最后一条语句,所以它又属于尾递归。35.1.2何时使用递归在以下三种情况下,常常要用到递归的方法。
1.定义是递归的
有许多数学公式、数列等的定义是递归的。例如,求n!和Fibonacci数列等。这些问题的求解过程可以将其递归定义直接转化为对应的递归算法。
42.数据结构是递归的
有些数据结构是递归的。例如,第2章中介绍过的单链表就是一种递归数据结构,其结点类型定义如下:
typedefstructLNode{ElemTypedata;structLNode*next; }LinkList;
该定义中,结构体LNode的定义中用到了它自身,即指针域next是一种指向自身类型的指针,所以它是一种递归数据结构。
5
对于递归数据结构,采用递归的方法编写算法既方便又有效。例如,求一个不带头结点的单链表L的所有data域(假设为int型)之和的递归算法如下:intSum(LinkList*L){if(L==NULL)return0;elsereturn(L->data+Sum(L->next));}63.问题的求解方法是递归的
有些问题的解法是递归的,典型的有Hanoi问题求解.该问题描述是:设有3个分别命名为X,Y和Z的塔座,在塔座X上有n个直径各不相同,从小到大依次编号为1,2,…,n的盘片,现要求将X塔座上的n个盘片移到塔座Z上并仍按同样顺序叠放,盘片移动时必须遵守以下规则:每次只能移动一个盘片;盘片可以插在X,Y和Z中任一塔座;任何时候都不能将一个较大的盘片放在较小的盘片上。设计递归求解算法。设Hanoi(n,x,y,z)表示将n个盘片从x通过y移动到z上,递归分解的过程是:7XYZ汉诺塔函数:Hanoi(intn,charx,chary,charz)作用:将n个圆盘从x座搬到z座,y作为过渡座。如:Hanoi(5,'X','Y','Z')8XYZ汉诺塔(1)先做:Hanoi(n-1,x,z,y)作用:将n-1个圆盘从x座搬到y座,z作为过渡座。9XYZ汉诺塔(1)先做:Hanoi(n-1,x,z,y)作用:将n-1个圆盘从x座搬到y座,z作为过渡座。10XYC汉诺塔(1)先做:Hanoi(n-1,x,z,y)作用:将n-1个圆盘从x座搬到y座,z作为过渡座。11XYZ汉诺塔(2)做:cout<<“将第”<<n<<“个盘片从”<<x<<“移到"<<z<<endl;作用:显示:将1个圆盘从x座搬到z座.12XYZ汉诺塔(2)做:cout<<“将第”<<n<<“个盘片从”<<x<<“移到"<<z<<endl;作用:显示:将1个圆盘从x座搬到z座.13XYZ汉诺塔(2)做:cout<<“将第”<<n<<“个盘片从”<<x<<“移到"<<z<<endl;作用:显示:将1个圆盘从x座搬到z座.14XYZ汉诺塔(3)做:Hanoi(n-1,y,x,z)作用:将n-1个圆盘从y座搬到z座,x作为过渡座。15XYZ汉诺塔(3)做:Hanoi(n-1,y,x,z)作用:将n-1个圆盘从y座搬到z座,x作为过渡座。16XYZ汉诺塔(3)做:Hanoi(n-1,y,x,z)作用:将n-1个圆盘从y座搬到z座,x作为过渡座。17XYZ汉诺塔Hanoi(n,x,y,z)Hanoi(n-1,x,z,y);move(n,x,z):将第n个圆盘从x移到z;Hanoi(n-1,y,x,z)18Hanoi问题的递归算法voidHanoi(intn,charx,chary,charz)//将n个盘片从x柱
//通过y柱移动到z柱{if(n==1)cout<<"将第"<<n<<"个盘片从"<<x<<“移到"<<z<<endl;else{Hanoi(n-1,x,z,y);cout<<"将第"<<n<<"个盘片从"<<x<<"移到"<<z<<endl;Hanoi(n-1,y,x,z);}}19intmain(){cout<<(“输入盘片个数:”);intn;cin>>n;Hanoi(n,‘X’,‘Y’,‘Z’);return0;}205.1.3递归模型
递归模型是递归算法的抽象,它反映一个递归问题的递归结构,例如,前面的递归算法对应的递归模型如下:
f(1)=1(1)f(n)=n*f(n-1)n>1(2)
其中,第一个式子给出了递归的终止条件,第二个式子给出了f(n)的值与f(n-1)的值之间的关系,我们把第一个式子称为递归出口,把第二个式子称为递归体。21
一般地,一个递归模型是由递归出口和递归体两部分组成,前者确定递归到何时结束,后者确定递归求解时的递推关系。递归出口的一般格式如下:
f(s1)=m1 (5.1)
这里的s1与m1均为常量,有些递归问题可能有几个递归出口。递归体的一般格式如下:
f(sn+1)=g(f(si),f(si+1),…,f(sn),cj,cj+1,…,cm) (5.2)
其中,n,i,j,m均为正整数。这里的sn+1是一个递归“大问题”,si,si+1,…,sn为递归“小问题”,cj,cj+1,…,cm是若干个可以直接(用非递归方法)解决的问题,g是一个非递归函数,可以直接求值。22
实际上,递归思路是把一个不能或不好直接求解的“大问题”转化成一个或几个“小问题”来解决,再把这些“小问题”进一步分解成更小的“小问题”来解决,如此分解,直至每个“小问题”都可以直接解决(此时分解到递归出口)。但递归分解不是随意的分解,递归分解要保证“大问题”与“小问题”相似,即求解过程与环境都相似。
23为了讨论方便,简化上述递归模型为:
f(s1)=m1 (5.3)f(sn)=g(f(sn-1),c) (5.4)求f(sn)的分解过程如下:
f(sn)↓f(sn-1)↓…↓f(s2)↓f(s1)24
一旦遇到递归出口,分解过程结束,开始求值过程,所以分解过程是“量变”过程,即原来的“大问题”在慢慢变小,但尚未解决,遇到递归出口后,便发生了“质变”,即原递归问题便转化成直接问题。上面的求值过程如下:
f(s1)=m1↓f(s2)=g(f(s1),c1)↓f(s3)=g(f(s2),c2)↓…↓f(sn)=g(f(sn-1),cn-1)25
这样f(sn)便计算出来了,因此,递归的执行过程由分解和求值两部分构成。265.2递归调用的实现原理用栈把本次调用的返回地址以及本次函数的参数值存储起来。每调用一次进栈一次,当返回时执行出栈处理。例:求阶乘的递归函数fun(n)的执行过程:n=5时intfun(intn){if(n==1) /*语句1*/return1; /*语句2*/else /*语句3*/return(fun(n-1)*n); /*语句4*/}27intfun(intn=5){if(n==1) /*语句1*/return1; /*语句2*/else /*语句3*/return(fun(n-1)*n); /*语句4*/}执行1,4句,由于第4句递归,中断地址d1:fun(n-1)和参数n=5入栈。然后调用fun(4).栈:d1528intfun(intn=4){if(n==1) /*语句1*/return1; /*语句2*/else /*语句3*/return(fun(n-1)*n); /*语句4*/}执行1,4句,由于第4句递归,中断地址d2:fun(n-1)和参数n=4入栈。然后调用fun(3).栈:d15d2429intfun(intn=3){if(n==1) /*语句1*/return1; /*语句2*/else /*语句3*/return(fun(n-1)*n); /*语句4*/}执行1,4句,由于第4句递归,中断地址d3:fun(n-1)和参数n=3入栈。然后调用fun(2).栈:d15d24d3330intfun(intn=2){if(n==1) /*语句1*/return1; /*语句2*/else /*语句3*/return(fun(n-1)*n); /*语句4*/}执行1,4句,由于第4句递归,中断地址d4:fun(n-1)和参数n=2入栈。然后调用fun(1).栈:d15d24d33d4231intfun(intn=1){if(n==1) /*语句1*/
return1; /*语句2*/else /*语句3*/return(fun(n-1)*n); /*语句4*/}执行1,2句,函数fun(1)=1结束,返回时,从栈退出栈顶元素获得返回地址d4,并将参数n恢复为2,继续执行函数fun(2)(从地址d4处继续)栈:d15d24d33d4232intfun(intn=2){if(n==1) /*语句1*/return1; /*语句2*/else /*语句3*/return(fun(n-1)*n); /*语句4*/}执行1,2句,函数fun(1)=1结束,返回时,从栈退出栈顶元素获得返回地址d4,并将参数n恢复为2,继续执行函数fun(2)(从地址d4处继续),其中fun(n-1)=1栈:d15d24d33d4233intfun(intn=2){if(n==1) /*语句1*/return1; /*语句2*/else /*语句3*/return(fun(n-1)*n); /*语句4*/}继续执行4句:return(1*2),函数fun(2)=2结束,返回时,从栈退出栈顶元素获得返回地址d3,并将参数n恢复为3,继续执行函数fun(3)(从地址d3处继续)栈:d15d24d3334intfun(intn=3){if(n==1) /*语句1*/return1; /*语句2*/else /*语句3*/return(fun(n-1)*n); /*语句4*/}继续执行4句:return(1*2),函数fun(2)=2结束,返回时,从栈退出栈顶元素获得返回地址d3,并将参数n恢复为3,继续执行函数fun(3)(从地址d3处继续),其中fun(n-1)=2栈:d15d24d3335intfun(intn=4){if(n==1) /*语句1*/return1; /*语句2*/else /*语句3*/return(fun(n-1)*n); /*语句4*/}继续执行4句:return(2*3),函数fun(3)=6结束,返回时,从栈退出栈顶元素获得返回地址d2,并将参数n恢复为4,继续执行函数fun(4)(从地址d2处继续)栈:d15d2436intfun(intn=5){if(n==1) /*语句1*/return1; /*语句2*/else /*语句3*/return(fun(n-1)*n); /*语句4*/}继续执行4句:return(2*3),函数fun(3)=6结束,返回时,从栈退出栈顶元素获得返回地址d2,并将参数n恢复为4,继续执行函数fun(4)(从地址d2处继续)栈:d15d2437intfun(intn=5){if(n==1) /*语句1*/return1; /*语句2*/else /*语句3*/return(fun(n-1)*n); /*语句4*/}继续执行4句:return(6*4),函数fun(4)=24结束,返回时,从栈退出栈顶元素获得返回地址d1,并将参数n恢复为5,继续执行函数fun(5)(从地址d1处继续)栈:d1538intfun(intn=5){if(n==1) /*语句1*/return1; /*语句2*/else /*语句3*/return(fun(n-1)*n); /*语句4*/}继续执行4句:return(24*5),函数fun(5)=120结束.栈:39求解fun(5)的过程如下:405.3递归算法的设计
递归的求解的过程均有这样的特征:先将整个问题划分为若干个子问题,通过分别求解子问题,最后获得整个问题的解。而这些子问题具有与原问题相同的求解方法,于是可以再将它们划分成若干个子问题,分别求解,如此反复进行,直到不能再划分成子问题,或已经可以求解为止。这种自上而下将问题分解、求解,再自上而下引用、合并,求出最后解答的过程称为递归求解过程。这是一种分而治之的算法设计方法。递归算法设计先要给出递归模型,再转换成对应的C/C++语言函数。41
递归设计的步骤如下:
(1)对原问题f(s)进行分析,假设出合理的“较小问题”f(s')(与数学归纳法中假设n=k-1时等式成立相似);
(2)假设f(s')是可解的,在此基础上确定f(s)的解,即给出f(s)与f(s')之间的关系(与数学归纳法中求证n=k时等式成立的过程相似);
(3)确定一个特定情况(如f(1)或f(0))的解,由此作为递归出口(与数学归纳法中求证n=1时等式成立相似)。42
例如,采用递归算法求实数数组A[0..n-1]中的最小值。
假设f(A,i)函数求数组元素A[0]~A[i]中的最小值。当i=0时,有f(A,i)=A[0];假设f(A,i-1)已求出,则f(A,i)=min(f(A,i-1),A[i]),其中min()为求两个值较小值函数。因此得到如下递归模型:
A[0] 当i=0时
f(A,i)=min(f(A,i-1),A[i]) 其他情况43由此得到如下递归求解算法:floatf(floatA[],inti){floatm;if(i==0)returnA[0];else{m=f(A,i-1); if(m>A[i])returnA[i]; elsereturnm; }}44例5.2:利用串的基本运算写出对串求逆的递归算法.解:设f(s)=s的逆串SqStringinvert(SqString&s){SqStrings1,s2;if(StrLength(s)>0){s1=invert(SubStr(s,2,StrLength(s)-1));s2=Concat(s1,SubStr(s,1,1));}elseStrCopy(s2,s);returns2;}45例5.3
求顺序表L的最大元素。ElemTypeMax(SqListL,inti,intj)//求第i~j元素中的最大值{intmid;ElemTypemax,max1,max2;if(i==j)max=L.data[i];else{mid=(i+j)/2;max1=Max(L,i,mid);//求第i~mid元素中的最大值
max2=Max(L,mid+1,j);//求第mid~j元素中的最大值
max=(max1>max2)?max1:max2;}returnmax;}46例5.4,设计不带头结点的单链表的相关递归算法a1a2an...Lf(L):大问题f(L->next):小问题47(1)求单链表中数据结点个数递归模型如下:f(L)=0 当L=NULLf(L)=f(L->next)+1 其他情况intcount(LinkList*L){if(L==NULL) return0;else returncount(L->next)+1;}递归算法如下:48(2)释放单链表中所有结点递归模型如下:f(L)=不做任何事情 当L=NULLf(L)=先做f(L->next);再释放*L结点 其他情况voidrelease(LinkList*L){if(L==NULL) return;else{release(L->next);
温馨提示
- 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
- 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
- 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
- 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
- 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
- 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
- 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。
最新文档
- 包过培训机构合同范本
- 劳工住宿合同范本
- 办公用品购置合同范本
- 共享菜园转让合同范本
- 公司外包收债合同范本
- 健康产业合同范本
- 农村修桥工程合同范本
- 2024年重庆松山医院招聘考试真题
- 写退货合同范本
- 2024年重庆市永川区三教镇招聘公益性岗位人员笔试真题
- 英语-广东省上进联考领航高中联盟2025届高三下学期开学考试题和答案
- 安全主任在2025年春季开学典礼上的讲话稿
- 2025届高考语文二轮复习语文备考策略
- 2025年春季新北师大版生物七年级下册全册教学课件
- 培训课件:律师客户沟通技巧
- 部编版语文小学二年级下册第一单元集体备课(教材解读)
- 房屋市政工程生产安全重大事故隐患判定标准(2024版)宣传画册
- 高等传热学全册课件
- (中外历史纲要下)历史 第三单元 大单元教学设计与单元评价
- 文华财经“麦语言”函数手册
- 苏教版科学2023四年级下册全册教案教学设计及反思
评论
0/150
提交评论