版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领
文档简介
第一章算法概述第二章递归与分治策略第三章动态规划第四章贪心算法第五章回朔法第六章分支限界法第七章概率算法算法设计与分析>目录12算法设计与分析>递归与分治2.1递归的概念直接或间接地调用自身的算法称为递归算法。由分治法产生的子问题往往是原问题的较小模式,为使用递归技术提供了方便。反复应用分治手段,可以使子问题与原问题类型一致而其规模却不断缩小,最终使子问题缩小到很容易直接求出其解,自然导致递归过程的产生。3算法设计与分析>递归与分治相关概念当子问题足够大,需要递归求解时,称为递归情况;当子问题足够小时,不再需要递归时,递归进入“触底”,进入基本情况;用函数自身给出定义的函数称为递归函数;递归式是一个等式或不等式,它通过更小的输入上的函数值来描述一个函数.4递归函数的内部执行过程
(1)运行开始时,为递归调用建立一个工作栈,其结构包括实参、局部变量和返回地址;(2)每次执行递归调用之前,把递归函数的实参和局部变量的当前值以及调用后的返回地址压栈;(3)每次递归调用结束后,将栈顶元素出栈,使相应的实参和局部变量恢复为调用前的值,然后转向返回地址指定的位置继续执行算法设计与分析>递归与分治5算法设计与分析>递归与分治例2-3Ackerman函数当一个函数及它的一个变量是由函数自身定义时,称这个函数是双递归函数。Ackerman函数A(n,m)定义如下:该变量是由函数自身定义6分析:A(n,m)的自变量m的每一个值都定义了一个单变量函数:M=0时,A(n,0)=n+2M=1时A(1,1)=2和A(n,1)=A(A(n-1,1),0)=A(n-1,1)+2,
故A(n,1)=2*n(上式是一个递推公式,每当n-1便加2)M=2时,A(1,2)=A(A(0,2),1)=A(1,1)=2A(n,2)=A(A(n-1,2),1)=2A(n-1,2),故A(n,2)=2n
。M=3时,类似的可以推出M=4时,A(n,4)的增长速度非常快,以至于没有适当的数学式子来表示这一函数。算法设计与分析>递归与分治7算法设计与分析>递归与分治1、证明A(1,1)=2因为n,m>=1使用递推式(4)得A(1,1)=A(A(0,1),0)由A(0,m)=1,故A(1,1)=A(1,0)由(1)式A(1,0)=2所以A(1,1)=22、证明A(n,1)=2n(n>=1)因为n,
m>=1使用递归式(4)得A(n,1)=A(A(n-1,1),0)由公式(3)A(n,1)=A(n-1,1)+2由此式看出m=1时,n每降一阶就增加2,一直降到n=1即A(1,1)共增加n个2,n个2相加和为2n所以A(n,1)=2n8算法设计与分析>递归与分治3、证明当m=2时A(n,2)=2n由递推式(4)A(n,2)=A(A(n-1,2),1),此时已经演变为m=1的情况,式中A(n-1,2)相当于2节中的n的位置,因此利用2节的结论有A(n,2)=2A(n-1,2)可以看出随着n的降阶会增长2的倍数,那么当n降为1时即A(1,2)的情况如何呢?根据递推式(4)有A(1,2)=A(A(0,2),1),再由公式(2)A(0,m)=1故A(1,2)=A(1,1)=2所以A(n,2)随着n的降阶每次乘2,共乘n次得A(n,2)=2n公式得证定义单变量的Ackerman函数A(n)为:A(n)=A(n,n)。定义其拟逆函数α(n)为:
α(n)=min{k|A(k)≥n}。
即α(n)是使n≤A(k)成立的最小的k值。α(n)在复杂度分析中常遇到。对于通常所见到的正整数n,有α(n)≤4。但在理论上α(n)没有上界,随着n的增加,它以难以想象的慢速度趋向正无穷大。算法设计与分析>递归与分治9递归树T(n)=3T(n/4)+O(n2)算法设计与分析>递归与分治10算法设计与分析>递归与分治1112补充:主方法(定理)求解这类递推式的方法是主方法,主方法依赖于下面的主定理,使用主定理可直接得到递推式的解。定理(主定理):a≥1且b>1是常数,f(n)是一个函数,T(n)由如下的递推式定义:T(n)=aT(n/b)+f(n),式中,n/b指n/b或n/b,则T(n)有如下的渐近界:(1)若对于某常数є>0,有f(n)=O(nlogba-є),则T(n)=(nlogba);(2)若f(n)=(nlogba),则T(n)=(nlogbalogn);(3)若对于某常数є>0,有f(n)=(nlogba+є),且对于某个常数c<1和所有足够大的n,有af(n/b)≤cf(n),则T(n)=(f(n))。T(n)=aT(n/b)+f(n)算法设计与分析>递归与分治13定理(主定理):a≥1且b>1是常数,f(n)是一个函数,T(n)由如下的递推式定义:T(n)=aT(n/b)+f(n),式中,n/b指n/b或n/b,则T(n)有如下的渐近界:(1)若对于某常数є>0,有f(n)=O(nlogba-є),则T(n)=(nlogba);
举例:T(n)=16T(n/4)+nT(n)=aT(n/b)+f(n)算法设计与分析>递归与分治14定理(主定理):a≥1且b>1是常数,f(n)是一个函数,T(n)由如下的递推式定义:T(n)=aT(n/b)+f(n),式中,n/b指n/b或n/b,则T(n)有如下的渐近界:(2)若f(n)=(nlogba),则T(n)=(nlogbalogn);
举例:T(n)=T(3n/7)+1T(n)=aT(n/b)+f(n)算法设计与分析>递归与分治15定理(主定理):a≥1且b>1是常数,f(n)是一个函数,T(n)由如下的递推式定义:T(n)=aT(n/b)+f(n),式中,n/b指n/b或n/b,则T(n)有如下的渐近界:(3)若对于某常数є>0,有f(n)=(nlogba+є),且对于某个常数c<1和所有足够大的n,有af(n/b)≤cf(n),则T(n)=(f(n))。
举例:T(n)=3T(n/4)+nlognT(n)=aT(n/b)+f(n)算法设计与分析>递归与分治递归小结优点:结构清晰,可读性强,而且容易用数学归纳法来证明算法的正确性,因此它为设计算法、调试程序带来很大方便。缺点:递归算法的运行效率较低,无论是耗费的计算时间还是占用的存储空间都比非递归算法要多。算法设计与分析>递归与分治16解决方法:在递归算法中消除递归调用,使其转化为非递归算法。1、采用一个用户定义的栈来模拟系统的递归调用工作栈。该方法通用性强,但本质上还是递归,只不过人工做了本来由编译器做的事情,优化效果不明显。2、用递推来实现递归函数。3、通过变换能将一些递归转化为尾递归,从而迭代求出结果。后两种方法在时空复杂度上均有较大改善,但其适用范围有限。算法设计与分析>递归与分治1718尾递归是在递归调用时“积累”之前调用的结果,并将其传入下一次递归调用中,将递归方法中的需要的“所有状态”通过方法的参数传入下一次调用中.算法设计与分析>递归与分治尾递归只会占用恒量的内存,形式上只要最后一个return语句是单纯函数就可以2.2分治法的基本思想算法设计与分析>递归与分治19将一个规模为n的问题分解为k个规模较小的子问题,这些子问题互相独立且与原问题相同。对这k个子问题分别求解。如果子问题的规模仍然不够小,则再划分为k个子问题,如此递归的进行下去,直到问题规模足够小,很容易求出其解为止。将求出的小规模的问题的解合并为一个更大规模的问题的解,自底向上逐步求出原来问题的解。20
子问题1的规模是n/2
子问题1的解
子问题2的解
子问题2的规模是n/2
原问题的解
原问题的规模是n算法设计与分析>递归与分治21算法设计与分析>递归与分治问题(N个输入)子问题1子问题2子问题K…子问题1子问题2子问题k…相同类型合并解分治法要求分解成同类子问题,并允许不断分解,使问题规模逐步减小,最终可用已知的方法求解足够小的问题。22算法设计与分析>递归与分治举例:二路归并排序分解过程:对任意长度为n的序列排序,首先将问题不断分解,直至问题可直接求解。长度为n的序列分解为2个n/2的子序列,依次循环,直至分解为n个长度为1的序列,显然长度为1的序列是有序表。合并过程:将已排序的的子序列不断合并,形成长度为n的排序序列。然后反复进行两两归并为n/2个有序表;再对n/2个有序表两两归并,循环直到得到长度为n的有序线性表。23算法设计与分析>递归与分治template<classType>voidMergeSort(Typea[],intleft,intright){if(left<right){
//至少有2个元素
inti=(left+right)/2;
//取中点
MergeSort(a,left,i);
//对前半部分排序
MergeSort(a,i+1,right);
//对后半部分排序
Merge(a,b,left,i,right);
//合并到数组b
Copy(a,b,left,right);
//复制回数组a
}}合并步,将有序表a[left,i]和a[i+1,right]合并到b[left,right]问题分解:规模为原来的一半;问题性质不变。24template<classType>voidMergeSort(Typea[],intleft,intright){if(left<right){
inti=(left+right)/2;
MergeSort(a,left,i);
MergeSort(a,i+1,right);
Merge(a,b,left,i,right);
Copy(a,b,left,right);
}}分析:二路归并排序的时空复杂性算法设计与分析>递归与分治时间复杂度T(n)cT(n/2)T(n/2)O(n)O(n)voidMerge(Typec[],Typed[],inti,intm,intr){intla,lb,lc;la=i;lb=m+1;lc=i;while(la<=m&&lb<=n){if(c[la]<c[lb])d[lc++]=c[la++];elsed[lc++]=c[lb++];}while(la<=m)d[lc++]=c[la++];
while(lb<=n)d[lc++]=c[lb++];}25二路归并排序算法存在以下递推式:解此递归方程,二路归并排序的时间复杂度是O(nlogn)。二路归并排序其空间复杂性为O(n)。算法设计与分析>递归与分治26算法设计与分析>递归与分治总结:分治法的适用条件分治法所能解决的问题一般具有以下特征:该问题规模缩小到一定的程度就可以容易地解决;该问题可以分解为若干个规模较小的相同子问题,即该问题具有最优子结构性质;利用该问题分解出的子问题的解可以合并为该问题的解;该问题所分解出的各个子问题是相互独立的,即子问题之间不包含公共的子问题。应用分治法的前提,它也是大多数问题可以满足的,此特征反映了递归思想的应用。关键特征,能否利用分治法完全取决于问题是否具有第四条特征,如果具备了第一条和第二条特征,而不具备第四条特征,则可以考虑贪心法或动态规划法。27算法设计与分析>递归与分治分治法的基本步骤:划分:把规模为n的原问题划分为k个规模较小的子问题,并尽量使这k个子问题的规模大致相同。求解子问题:各子问题的解法与原问题的解法通常是相同的,可以用递归的方法求解各个子问题,有时也可用循环来实现。合并:把各个子问题的解合并起来。divide-and-conquer(P){
if(|P|<=n0)adhoc(P);//解决小规模的问题
dividePintosmallersubinstancesP1,P2,...,Pk;//分解问题
for(i=1,i<=k,i++)yi=divide-and-conquer(Pi);//递归的解各子问题
returnmerge(y1,...,yk);//将各子问题的解合并为原问题的解
}分治法的基本步骤算法设计与分析>递归与分治28在用分治法设计算法时,最好使子问题的规模大致相同,即,将一个问题分成大小相等的k个子问题(许多问题取k=2)。使子问题规模大致相等的做法是出自一种平衡子问题的思想,这通常比子问题规模不等的做法要好。|P|问题P的规模n0为阀值adhoc(P)基本子算法29分治法的复杂性分析从一般设计模式看,用分治法设计的程序通常是一个递归算法。分治法将规模为n的问题分成k个规模为n/m的子问题:设n0=1,且adhoc解问题耗费1个单位时间。再设将原问题分解为k个子问题以及用merge将k个子问题的解合并为原问题的解需用f
(n)个单位时间。用T(n)(递归方程)表示该分治法求解规模为|P|=n的问题所需的计算时间:算法设计与分析>递归与分治30通过迭代法求得方程解:基本子问题花费时间合并步花费时间
算法设计与分析>递归与分治312.3二分搜索技术算法设计与分析>递归与分治问题描述:已知一个按非降次序排列的元素表a1,a2,…,an,判定某个给定元素x是否在该表中出现,若是,则找出该元素在表中的位置,并置于j,否则,置j为0。一般解决方法:从头到尾查找一遍。a1a2a3a4…anx…成功与不成功的最坏情况下需O(n)次比较分析:如果n=1即只有一个元素,则只要比较这个元素和x就可以确定x是否在表中。因此这个问题满足分治法的第一个适用条件分析:比较x和a的中间元素a[mid],若x=a[mid],则x在L中的位置就是mid;如果x<a[mid],由于a是递增排序的,因此假如x在a中的话,x必然排在a[mid]的前面,所以我们只要在a[mid]的前面查找x即可;如果x>a[i],同理我们只要在a[mid]的后面查找x即可。无论是在前面还是后面查找x,其方法都和在a中查找x一样,只不过是查找的规模缩小了。这就说明了此问题满足分治法的第二个和第三个适用条件。
分析:很显然此问题分解出的子问题相互独立,即在a[i]的前面或后面查找x是独立的子问题,因此满足分治法的适用条件。分析:该问题的规模缩小到一定的程度就可以容易地解决;该问题可以分解为若干个规模较小的相同问题;分解出的子问题的解可以合并为原问题的解;分解出的各个子问题是相互独立的。算法设计与分析>递归与分治32二分搜索算法:template<classType>intBinarySearch(Typea[],constType&x,intl,intr){while(r>=l){intm=(l+r)/2;if(x==a[m])returnm;if(x<a[m])r=m-1;elsel=m+1;}return-1;}算法复杂度分析:每执行一次算法的while循环,待搜索数组的大小减少一半。在最坏情况下,while循环被执行O(logn)次,循环体内运算需要O(1)时间。算法在最坏情况下的计算时间复杂性为O(logn)。算法设计与分析>递归与分治332.8快速排序[算法思路]对A[p:r],按以下步骤进行排序:(1)分解:取A中一元素为支点,将A[p:r]分成3段:A[p:q-1],A[q],A[q+1:r],其中A[p:q-1]中元素A[q],A[q+1:r]中元素A[q];(2)求解:通过递归调用分别对A[p:q-1]和A[q+1:r]排序。(3)合并:合并A[p:q-1],A[q],A[q+1:r]为A[p:r]。34算法设计与分析
>
递归与分治
>快速排序
得:Tmax(n)=O(n2)[快速排序算法]template<classT>voidQuickSoft(T
a[],intp,intr){if(p<r){intq=Partition(a,p,r)QuickSort(a,p,q-1);//对左半段排序
QuickSoft(a,q+1,r);//对右半段排序template<classT>intPartion(Ta[],intp,intr){inti=p;j=r+1;tx=a[p];//取支点
//将x的元素交换到左边
//将x的元素交换到右边
while(true){while(a[++i]<x);
while(a[--j]>x);
if(i>=j)break;swap(a[i],a[j]);}a[p]=a[j];
a[j]=x;returnj}
Tmin(n)=O(1)n<=12T(n/2)+O(n)n>1
得:Tmin(n)=O(nlogn)[复杂性分析]
Tmax(n)=O(1)n<=1T(n-1)+O(n)n>135[随机选择支点的快速排序]template<classT>voidrandomizedQuickSoft(Ta[],intp,intr){if(p<r){intq=randomizedPartition(a,p,r)randomizedQuickSort(a,p,q-1);//对左半段排序
randomizedQuickSoft(a,q+1,r);//对右半段排序
}}算法设计与分析>
递归与分治>快速排序template<classT>intrandomizedPartition(Ta[],intp,intr){inti=random(p,r)swap(a[i],a[p])returnPartition(a,p,r)36快速排序算法性能取决于划分的对称性2.9线性时间选择[问题陈述]
求n个元素中的第k小元素.给定线性序集中n个不同的元素和一个整数k,1≼k≼n要求找出这n个元素中第k小的元素,即如果将这n个元素依其线性序排列时,排在第k个位置的元素即为我们要找的元素。当k=1时,找最小元;当k=n时,找最大元素;当k=(n+1)/2时,找中位数.37算法设计与分析>
递归与分治38算法设计与分析>
递归与分治[算法思路]模仿快速排序,对输入数组A进行二分,使子数组A1的元素
A2中的元素,分点J由随机数产生.若KJ,则K为A1的第K小元,若K>J,则K为A2的第K-J小元.39算法设计与分析>
递归与分治找数组a[0:n-1]第k小元素调用RandomizedSelect(a,0,n-1,k)template<classT>TRandomizedSelect(a[],intp,intr,intk){if(p==r)returna[p];
inti=RandomizedPartition(a,p,r);j=i-p+1;
//支点元素是第j小个元素,左半部元素个数
ifk==jreturnA[i];
if(k<j)returnRandomizedSelect(a,p,i-1,k);else
returnRandomizedSelect(a,i+1,r,k-j);
}p,r数组范围K:第k小个元素与快速排序算法不同,它只对子数组之一进行递归处理。40执行RandomizedPartiton,数组a[p:r]划分成两个子数组a[p:i]和a[i+1:r],其中a[p:i]元素≤a[i+1:r]元素计算子数组a[p:i]中元素个数j如果k<j,则a[p:i]中第k小元素落在子数组a[p:i]中;如果k>j,则要找的第k小元素落在子数组a[i+1:r]中,要找的a[p:r]中第k小的元素是a[i+1:r]中的第k-j小的元素。如果k=j,找到第k小个元素,返回a[i]递归循环。算法设计与分析>
递归与分治41
Tmin(n)=dT(n)=T(n/2)+cn[分析]
最坏情况:Tmax(n)=
(n2)(左半总为空)(等分点)
得:Tmin(n)=(n)算法设计与分析>
递归与分治最坏情况下的选择算法分析42算法设计与分析>
递归与分治算法RandomizedSelect需要Ω(n²)计算时间。例如在找最小元素时,总是在最大元素处划分。(n-1)+(n-2)+…+1=n²算法RandomizedSelect不稳定的原因:由于随机划分函数使用了一个随机数产生器Random,产生p和r之间的一个随机整数,因此产生的划分基准是随机的。可以证明,算法可以在O(n)平均时间内找出n个输入元素中的第k小元素。43[算法思路](中间的中间):将A中元素分为n/r组,除最后一组外,每组元素为r个,用任意排序算法排序,取出每组的中位数,对n/r个中位数递归调用,找到中位数,以该元素作为划分基准.算法设计与分析>
递归与分治xtemplate<classT>TSelect(Ta[],intp,intr,intk){if(r-p<75){用简单排序法排序;
returna[p+k-1]};
for(inti=0;i<=(r-p-4)/5;i++)
将a[p+5*i]至a[p+5*i-4]的第3小元素与a[p+i]交换位置;
//找中位数的中位数
Tx=Select(a,p,p+(r-p-4)/5,(r-p-4)/10);inti=Partition(a,p,r,x),j=i-p+1;if(k<=j)returnSelect(a,p,i,k)elsereturnSelect(a,i+1,r,k-j)}最坏情况下的选择算法:[算法分析]
T(n)=得:T(n)=O(n)c1n<75c2n+T(n/5)+T(3n/4)定理:若r=5且数组中元素互不相同,则n>75时,max{|left|,|right|}<=3n/4442.4大整数的乘法[问题描述]设X,Y是两个n位二进制数,求X*Y.45算法设计与分析>递归与分治一般方法:
O(n2)
效率太低!分治法:复杂性分析:
n/2位n/2位n/2位n/2位X=
Y=ABCDT(n)=O(n2)没有改进46[分析]这两个算式更复杂一些,但它们仅需要3次n/2位乘法。{ac、bd和(a±b)(d±c)}[算法改进]算法设计与分析>递归与分治为了降低时间复杂度,必须减少乘法的次数。为此,把XY写成另外的形式:XY=ac2n+((a-b)(d-c)+ac+bd)2n/2+bd或XY=ac2n+((a+b)(c+d)-ac-bd)2n/2+bd结论:T(n)=O(nlog3)=O(n1.59)较大的改进{S=SIGN(X)*SIGN(Y);
X:=ABS(X);Y:=ABS(Y);
ifn=lthenif(X=1)and(Y=1)thenreturn(S)elsereturn(0)else{A:=X的左边n/2位;
B:=X的右边n/2位;
C:=Y的左边n/2位;
D:=Y的右边n/2位;
m1:=MULT(A,C,n/2);
m2:=MULT(A-B,D-C,n/2);
m3:=MULT(B,D,n/2);
S:=S*(m1*2n+(m1+m2+m3)*2n/2+m3);
return(S)}}{X,Y为2个小于2n的二进制整数,S=XY}functionMULT(X,Y,n)大数相乘的分治递归算法二进制大整数乘法同样可应用于十进制大整数的乘法.存放XY的符号存放三个乘积算法472.5Strassen矩阵相乘法[常规算法]设矩阵A=(aij)nn,B=(bij)nn,C=AB=(cij)nn
计算C共需nn2个乘法,n2(n-1)个加法.
T(n)=O(n3)
算法设计与分析>
递归与分治48如果依此定义来计算矩阵A和B的乘积矩阵C,则每计算C的一个元素C[i,j],需要做n次乘法和n-1次加法。49算法设计与分析>
递归与分治C11=A11B11+A12B21C12=A11B12+A12B22C21=A21B11+A22B21C22=A21B12+A22B22首先假定n是2(n=2k)的幂,如果相乘的两矩阵A和B不是方阵,可以通过适当添加全零行和全零列,使之成为行列数为2的幂的方阵。使用分治法,将矩阵A、B和C中每一矩阵都分块成4个大小相等的子矩阵,每个子矩阵都是n/2×n/2的方阵。由此可将方程C=A×B重写为:[算法思路]
50算法设计与分析>
递归与分治如果n=2,则两个2阶方阵的乘积可以直接计算出来,共需8次乘法和4次加法。当子矩阵的阶大于2时,可以继续将子矩阵分块,直到子矩阵的阶降为2。这样,就产生了一个分治降阶的递归算法。依此算法,计算2个n阶方阵的乘积转化为计算8个n/2阶方阵的乘积和4个n/2阶方阵的加法。2个n/2×n/2矩阵的加法显然可以在c*n2/4(O(n2))时间内完成,这里c是一个常数。上述分治法的计算时间耗费T(n)的递归方程满足:T(n)=O(n3)没有改进,原因是没有减少矩阵乘法次数。51算法设计与分析>
递归与分治为了降低时间复杂度,必须减少乘法次数,其关键在于计算2个2阶方阵的乘积时所用乘法次数能否少于8次。为此,Strassen提出了一种只用7次乘法运算计算2阶方阵乘积的方法(但增加了加/减法次数):M1=A11(B12-B22),
M2=(A11+A12)B22M3=(A21+A22)B11,
M4=A22(B21-B11)M5=(A11+A22)(B11+B22),M6=(A12-A22)(B21+B22)M7=(A11-A21)(B11+B12)C11=M5+M4-M2+M6C12=M1+M2C21=M3+M4C22=M5+M1-M3-M7算法设计与分析>
递归与分治
得:T(n)=O(nlog7)=O(n2.81)[分析]
52做了这7次乘法后,再做若干次加/减法就可以得到:较大的改进procedureSTRASSEN(n,A,B,C);
{ifn=2thenMATRIX_MULTIPLY(A,B,C)else{将矩阵A和B分块;
STRASSEN(n/2,A11,B12-B22,M1);
STRASSEN(h/2,A11+A12,B22,M2);
STRASSEN(n/2,A21+A22,B11,M3);
STRASSEN(n/2,A22,B21-B11,M4);
STRASSEN(n/2,A11+A22,B11+B22,M5)STRASSEN(n/2,A12-A22,B21+B22,M6);
STRASSEN(n/2,A11+A21,B11+B12,M7)C:=矩阵相乘Strassen算法算法算法设计与分析>
递归与分治53[问题陈述]在一个2k2k个方格组成的棋盘中,恰有一方格残缺,残缺方格的位置有22k种。故对任何k≥0,残缺棋盘有22k种.要求用L型骨牌覆盖残缺棋盘上的所有方格且任何2个L型骨牌不得重叠覆盖。2.6棋盘覆盖算法设计与分析>
递归与分治54552k2k
的棋盘覆盖中,用到的骨牌数为(4k-1)/3算法设计与分析>
递归与分治[算法思路]
当k>0时,将2k2k棋盘分割为4个2k-12k-1子棋盘,残缺方格必位于4个子棋盘之一.其余3个子棋盘中无残缺方格.
用一个L型骨牌覆盖这3个较小棋盘的结合处.这3个子棋盘上被L型骨牌覆盖的方格就成为该棋盘上的残缺方格,原问题转化为4个较小规模的棋盘覆盖问题。递归地使用这种分割,直至棋盘简化为11棋盘.算法设计与分析>
递归与分治为此将剩余3棋盘转化为残缺棋盘:
56[算法分析]
设T(k)为覆盖2k2k残缺棋盘的时间,
当k=0时覆盖它需要常数时间O(1).当k>0时,测试哪个子棋盘残缺以及形成3个残缺子棋盘需要O(1),覆盖4个残缺子棋盘需四次递归调用,共需时间4T(k-1).
T(k)=O(1)4T(k-1)+O(1)
迭代法解得:T(k)=(4k)与所需骨牌数同阶算法设计与分析>
递归与分治57voidChessBoard(inttr,inttc,intdr,intdc,intsize){if(size==1)return;
intt=tile++,//L型骨牌数
s=size/2;//分割棋盘
//覆盖左上角子棋盘
if(dr<tr+s&&dc<tc+S)
//特殊方格在此棋盘中
ChessBoard(tr,tc,dr,dc,S);
else{//此棋盘中无特殊方格
//用t号L型骨牌覆盖右下角
Board[tr+s-1][tc+s-1]=t;
//覆盖其余方格
Chessboard(tr,tc,tr+s-1,tc+s-l,s);}算法设计与分析>
递归与分治tr:左上角方格行tc:左上角方格列dr:残缺方格行dc:残缺方格列size:棋盘行数Board:为全局变量二维数组表示棋盘58
//覆盖右上角子棋盘
if(dr<=tr+s&&dc>=tc+S)
//特殊方格在此棋盘中
ChessBoard(tr,tc+s,dr,dc,S);
else{//此棋盘中无特殊方格
//用t号骨牌覆盖左下角
Board[tr+s-1][tc+s]=t;
//覆盖其余方格
Chessboard(tr,tc+s,tr+s-1,tc+s,s);}
………………...voidoutputBoard(intsize){forinti=0;i<size;i++){forintj=0;j<size;j++);
cout<<setw(5)<<board[i][j];cout<<endl;}}算法设计与分析>
递归与分治5960算法设计与分析>
递归与分治给定平面S上n个点,找其中的一对点,使得在
n(n-1)/2个点对中,该点对的距离最小。2.10最接近点对问题[问题描述]简化问题(一维问题)为了使问题易于理解和分析,先来考虑一维的情形。此时,S中的n个点退化为x轴上的n个实数x1,x2,…,xn。最接近点对即为这n个实数中相差最小的2个实数。61算法设计与分析>
递归与分治假设我们用x轴上某个点m将S划分为2个子集S1和S2,基于平衡子问题的思想,用S中各点坐标的中位数来作分割点。递归地在S1和S2上找出其最接近点对{p1,p2}和{q1,q2},并设d=min{|p1-p2|,|q1-q2|},S中的最接近点对或者是{p1,p2},或者是{q1,q2},或者是某个{p3,q3},其中p3∈S1且q3∈S2。能否在线性时间内找到p3,q3?62算法设计与分析>
递归与分治如果S的最接近点对是{p3,q3},即|p3-q3|<d,则p3和q3两者与m的距离不超过d,即p3∈(m-d,m],q3∈(m,m+d]。由于在S1中,每个长度为d的半闭区间至多包含一个点(否则必有两点距离小于d),并且m是S1和S2的分割点,因此(m-d,m]中至多包含S中的一个点。由图可以看出,如果(m-d,m]中有S中的点,则此点就是S1中最大点。同理,如果(m,m+d]中有S中的点,则此点就是S2中最小点。用线性时间就能找到区间(m-d,m]和(m,m+d]中所有点,即p3和q3。用线性时间就可以将S1的解和S2的解合并成为S的解。能否在线性时间内找到p3,q3?63算法设计与分析>
递归与分治BoolCpairl(S,d){n=|S|;if(n<2){d=MAX;returnfalse;}m=S中各点坐标的中位数;
Cpairl(S1,d1);
//找到S1中最短距离
Cpairl(S2,d2);
//找到S2中最短距离p=max(S1);q=min(S2);
//找到S1和S2中相邻接点
d=min(d1,d2,q-p);returntrue;}
T(n)=O(1)T(n)=2T(n/2)+O(n)
得:T(n)=O(nlogn)
1)n较小时直接求(n=2).2)将S上的n个点分成大致相等的2个子集S1和S23)分别求S1和S2中的最接近点对
4)求一点在S1、另一点在S2中的最近点对
5)从上述三对点中找距离最近的一对.算法设计与分析>
递归与分治[算法思路]64二维平面
温馨提示
- 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
- 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
- 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
- 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
- 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
- 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
- 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。
最新文档
- GB/T 22142-2024饲料添加剂有机酸通用要求
- 2024年度生态环境保护责任担保合同
- 安全用电施工协议书模板6
- 2024年度船舶租赁合同:航务管理与运营
- 2024年度企业信息系统集成服务合同
- 2024年度社交电商平台商家入驻合同:平台与商家之间的经营规范和费用结算3篇
- 2024年度电影项目评估与审计合同
- 《对数函数课时》课件
- 2024年度体育用品销售区域运营合同
- 2024年度国际汽车销售与维修服务合同2篇
- 高考生物一轮复习必修1第四单元细胞的生命历程第17讲减数分裂与有丝分裂的比较学案
- 叙事护理案例分享演讲课件
- 疾控中心:常见传染病防治手册
- (完整版)三大改造课件
- 无人机飞行控制系统设计(山东联盟)智慧树知到期末考试答案章节答案2024年山东航空学院
- 土地复垦施工组织设计58446
- 《2.6蚕的一生》说课稿、教案、教学设计与同步练习
- 路缘石安装安全技术交底
- 急性胰腺炎的诊断与处理:国内外主要指南的比较与解读
- 第12课《终身学习 持续发展》第2框《信息素养助力发展》-【中职专用】《心理健康与职业生涯》同步课堂课件
- 《高分子物理》课件-晶态高聚物应力应变曲线第三阶段
评论
0/150
提交评论