1、第6章 动态规划动态规划是我们要介绍的几种算法设计方法中难度最大的一种,它建立在最优原则的基础上。采用动态规划方法,可以高效地解决许多用贪婪算法或分而治之算法无法解决的问题。在介绍动态规划的原理之后,将分别考察动态规划方法在解决背包问题、图象压缩、矩阵乘法链、最短途径、无交叉子集等方面的应用。1算法思想和贪婪算法同样,在动态规划中,可将一个问题的解决方案视为一系列决策的结果。不同的是,在贪婪算法中,每采用一次贪婪准则便做出一个不可撤回的决策,而在动态规划中,还要考察每个最优决策序列中是否包含一个最优子序列。例6-1最短路经考察下图中的有向图。假设要寻找一条从源节点s=1到目的节点d=5的最短途
2、径,即选择此途径所通过的各个节点。第一步可选择节点2,3或4。假设选择了节点3,则此时所规定解的问题变成:选择一条从3到5的最短途径。假如3到5的途径不是最短的,则从1开始通过3和5的途径也不会是最短的。所以在最短途径问题中,假如在的第一次决策时到达了某个节点v,那么不管v是如何拟定的,此后选择从v到d的途径时,都必须采用最优策略。例6-20/1背包问题考察前面的0/1背包问题。如前所述,在该问题中需要决定x1 xn的值。假设按i=1,2,n的顺序来拟定xi的值。假如置x1=0,则问题转变为相对于其余物品(即物品2,3,n),背包容量仍为c的背包问题。若置x1=1,问题就变为关于最大背包容量为
3、c- w1的问题。现设rc,c-w1为剩余的背包容量。在第一次决策之后,剩下的问题便是考虑背包容量为r时的决策。不管x1是0或是1,x2,xn必须是第一次决策之后的一个最优方案,假如不是,则会有一个更好的方案y2,yn,因而x1,y2,yn是一个更好的方案。假设n=3,w=100,14,10,p=20,18,15,c=116。若设x1=1,则在本次决策之后,可用的背包容量为r=116-100=16。x2,x3=0,1符合容量限制的条件,所得值为15,但由于x2,x3=1,0同样符合容量条件且所得值为18,因此x2,x3=0,1并非最优策略。即x=1,0,1可改善为x=1,1,0。若设x1=0,
4、则对于剩下的两种物品而言,容量限制条件为116。总之,假如子问题的结果x2,x3不是剩余情况下的一个最优解,则x1,x2,x3也不会是总体的最优解。例6-3航费假设某航线价格表为:从亚特兰大到纽约或芝加哥,或从洛杉矶到亚特兰大的费用为$100;从芝加哥到纽约票价$20;而对于路经亚特兰大的旅客,从亚特兰大到芝加哥的费用仅为$20。从洛杉矶到纽约的航线涉及到对中转机场的选择。洛杉矶芝加哥亚特兰大纽约1001002010020假如问题状态的形式为(起点,终点),那么在选择从洛杉矶到亚特兰大后,问题的状态变为(亚特兰大,纽约)。从亚特兰大到纽约的最便宜航线是从亚特兰大直飞纽约,票价$100。而使用直
5、飞方式时,从洛杉矶到纽约的花费为$200。但是,从洛杉矶到纽约的最便宜航线为洛杉矶-亚特兰大-芝加哥-纽约,其总花费为$140(在解决局部最优途径亚特兰大到纽约过程中选择了最低花费的途径:亚特兰大-芝加哥-纽约)。假如用三维数组(tag,起点,终点)表达问题状态,其中tag为0表达转飞,tag为1表达其他情形,那么在到达亚特兰大后,状态的三维数组将变为(0,亚特兰大,纽约),它相应的最优途径是经由芝加哥的那条途径。当最优决策序列中包含最优决策子序列时,可建立动态规划递归方程通常是根据最优准则,得出动态规划计算的“递归”计算式。通常,这个递归式也是解决问题的关键,写出递归公式,然后用递归程序实现
6、,并消除反复计算在大多数情况下,求出递归方程是实际工作中的难点而消除反复计算则成为问题是否可以实际求解的关键(dynamic-programming recurrence equation),它可以帮助我们高效地解决问题。例6-40/1背包在例6-2的0/1背包问题中,最优决策序列由最优决策子序列组成。假设f(i,y)先把这个表达的地意义弄清楚?表达例6-2中剩余容量为y,剩余物品为i,i+1,n时的最优解的值,即:这个递归方程事实上将所有的情形所有考虑进去了,然后再从中找到一个最优解运用最优序列由最优子序列构成的结论,可得到f的递归式。f(1,c)是初始时背包问题的最优解。可通过递归或迭代来
7、求解f(1,c)。从f(n,*)开始迭式,先求出f(n,*),然后递归计算f(i,*)(i=n-1,n-2,.,2),最后得出f(1,c)。对于例6-2(w=100,14,10,p=20,18,15)可得:f(3,y)=0 0y10f(3,y)=15 y10f(2,y)=0 0y10f(2,y)=15 10y14f(2,y)=18 14y24 和f(2,y)=33 y24剩2、3两个物品的情况因此最优解:f(1,116)= maxf(2,116),f(2,116-w1)+p1= maxf(2,116),f(2,16)+20= max33,38 = 38。现在计算xi值,环节如下:若f(1,c)
8、=f(2,c),则x1=0,否则x1=1。接下来需从剩余容量c-w1中寻求最优解,用f(2,c-w1)表达最优解。依此类推,可得到所有的xi(i=1n)值。在该例中,可得出f(2,116)=33f(1,116),所以x1=1。接着运用返回值38-p1=18计算x2及x3,此时r=116-w1=16,又由f(2,16)=18,得f(3,16)=14f(2,16),因此x2=1,此时r=16-w2=2,所以f(3,2)=0,即得x3=0。动态规划方法采用最优原则(principle of optimality)来建立用于计算最优解的递归式。所谓最优原则即不管前面的策略如何,此后的决策必须是基于当前
9、状态(由上一次决策产生)的最优决策。由于对于有些问题的某些递归式来说并不一定能保证最优原则,因此在求解问题时有必要对它进行验证。若不能保持最优原则,则不可应用动态规划方法。在得到最优解的递归式之后,需要执行回溯(traceback)以构造最优解。编写一个简朴的递归程序来求解动态规划递归方程是一件很诱人的事。然而,正如我们将在下文看到的,假如不努力地去避免反复计算,递归程序的复杂性将非常可观。假如在递归程序设计中解决了反复计算问题时,复杂性将急剧下降。动态规划递归方程也可用迭代方式来求解,这时很自然地避免了反复计算。事实上,这也是动态规划求解的两个基本方法:1、 消除反复的递归2、 采用迭代来代
10、替递归尽管迭代程序与避免反复计算的递归程序有相同的复杂性,但迭代程序不需要附加的递归栈空间,因此将比避免反复计算的递归程序更快。2应用2.1 0/1背包问题背包问题的递归方程可以用几种方法具体的软件实现,即前面的递归公式求解,还是一个比较复杂的问题求得。1)递归策略在例6-4中已建立了背包问题的动态规划递归方程,求解递归式(6-2)的一个很自然的方法便是使用程序6-1中的递归算法。该模块假设p、w和n为输入,且p为整型,F(1,c)返回f(1,c)值。程序6-1背包问题的递归函数实际程序中,w和p以及n需要“输入”到递归函数中消除反复计算,假如y是整数,则可以考虑用一个二维数组保存F(i,y)
11、,从而消除反复计算int F(int i,int y)/返回f(i,y).if(i=n) return (ywn)?0:pn;if(ywi) return F(i+1,y);elsereturn max(F(i+1,y),F(i+1,y-wi)+pi);程序6-1的时间复杂性为:t(n)=(2n)。在所有f(i,y)计算完毕后,再用下列程序计算xitemplatevoid Traceback(T *f,int w,int c,int n,int x)/计算xfor(int i=1;in;i+)if(fic=fi+1c) xi=0;else xi=1; c-=wi; xn=(fnc)?1:0;例
12、6-5设n=5,p=6,3,5,4,6,w=2,2,6,5,4且c=10,求f(1,10)。为了拟定f(1,10),调用函数F(1,10)。递归调用的关系如图6-1的树型结构所示。每个节点用y值来标记。对于第j层的节点有i=j,因此根节点表达F(1,10),而它有左孩子和右孩子,分别相应F(2,10)和F(2,8)。总共执行了28次递归调用。但我们注意到,其中也许具有反复前面工作的节点,如f(3,8)计算过两次,相同情况的尚有f(4,8)、f(4,6)、f(4,2)、f(5,8)、f(5,6)、f(5,3)、f(5,2)和f(5,1)。假如保存以前的计算结果,则可将节点数减至19,由于可以丢弃
13、图中的阴影节点。图6-1 递归调用树正如在例6-5中所看到的,程序6-1做了一些不必要的工作。为了避免f(i,y)的反复计算,必须定义一个用于保存已被计算出的f(i,y)值的表格L,该表格的元素是三元组(i,y,f(i,y)。在计算每一个f(i,y)之前,应检查表L中是否已包含一个三元组(i,y,*),其中*表达任意值。假如已包含,则从该表中取出f(i,y)的值,否则,对f(i,y)进行计算并将计算所得的三元组(i,y,f(i,y)加入表L。L既可以用散列的形式存储,也可用二叉搜索树的形式存储。2)权为整数的迭代方法当权即 w为整数时,可设计一个相称简朴的算法(称为Knapsack算法,见程序
14、6-2)来求解f(1,c)。该算法基于例6-4所给出的策略,因此每个f(i,y)只计算一次。为简朴起见,程序6-2用二维数组f来保存各f的值。而回溯函数Traceback用于拟定由程序6-2所产生的xi值。函数Knapsack的复杂性为(nc),而Traceback的复杂性为(n)。程序6-2 f和x的迭代计算下面程序中,要注意二维数组的真正实现,c语言中二维数组的传递需要特别注意注意下列三种表达方式的差异:T fN T *fT *fNtemplateT max( T x, T y )return( (xy)?x:y );/ 注意规定yMax用宏定义,大于ctemplatevoid Knaps
15、ack( T p,int w,int c,int n,T (*f)yMax )/对于所有i和y计算fiyint i, y;/初始化ffor( y=0;yyMax;y+)fny=0;for( y=wn;y1;i-)for( y=0;yyMax;y+)fiy=fi+1y;for( y=wi;y=w1)f1c=max(f1c,f2c-w1+p1);templatevoid Traceback(T (*f)yMax,int w,int c,int n,int x)/计算xint i;for( i=1;in;i+)if(fic=fi+1c) xi=0;else xi=1; c-=wi; xn=(fnc)
16、?1:0;程序的具体实现如下(为简化起见,采用整型变量,同时,由于二维数组的传递比较麻烦,将数组f作为整形全局数组):#include const yMax=20;/ 规定比c大const n=5;/ 物体数量int fn+1yMax;int max( int x, int y )return( (xy)?x:y );void Knapsack(int p,int w,int c)/对于所有i和y计算fiy/初始化fnint i, y;for( y=0;yyMax;y+)fny=0;for( y=wn;y1;i-)for( y=0;yyMax;y+)fiy=fi+1y;for( y=wi;y=
17、w1)f1c=max(f1c,f2c-w1+p1);void Traceback(int w,int c,int x)/计算xint i;for( i=1;in;i+)if(fic=fi+1c) xi=0;else xi=1; c-=wi; xn=(fnc)?1:0;void main()int w=0,2,2,6,4,5, p=0,6,3,5,4,6, xn+1;int i,c=10;Knapsack( p, w, c);Traceback( w, c, x);for( i=1; i=n; i+ )printf( %d , xi );程序有两个缺陷:l 规定权为整数;l 当背包容量c很大时,
18、程序6-2的速度慢于程序6-1。由于实际的计算中,f中的很多元素见前面实例计算 F(1,116)按6-2 需要计算3*116个元素,而实际只用到其中大约7个元素是多余计算的。一般情况下,若c2n,程序6-2的复杂性为W (n2n)。作为一个思考题,希望大家考虑,假如c特别大,为消除反复计算二引入的二维数组f就特别大,如何来解决存储问题,一般可以考虑不用数组,采用其他手段来存,如前面所述的三元组或者稀疏矩阵等。2.2图像(字节流)压缩下列方法实际也用于其他数据的压缩。数字化图像是mm的像素阵列。假定每个像素有一个0255的灰度值。因此存储一个像素至多需8位。若每个像素存储都用最大位8位,则总的存
19、储空间为8m2位。为了减少存储空间,我们将采用变长模式(variable bit scheme),即不同像素用不同位数来存储。像素值为0和1时只需1位存储空间;值2、3各需2位;值4,5,6和7各需3位;以此类推,使用变长模式的环节如下:l 图像线性化 根据图6-3a中的折线将mm维图像转换为1m2维矩阵。图6-3 数字图像a)“之”形的行主顺序 b)灰度值l 分段 将像素组提成若干个段,分段原则是:每段中的像素位数相同。每个段是相邻像素的集合且每段最多含256个像素,因此,若相同位数的像素超过256个的话,则用两个以上的段表达。l 创建文献 涉及三个部分:SegmentLength,Bits
20、PerPixel和Pixels。第一个部分为所建的段的长度-1,其中各项均为8位长。第二部分:BitsPerPixel为各段中每个像素的存储位数-1,其中各项均为3位。第三部分Pixels则是以变长格式存储的像素的二进制串。l 压缩文献 压缩所建立的文献,以减少空间需求。上述压缩方法的效率(用所得压缩率表达)很大限度上取决于长段的出现频率。例6-8考察图6-3b的44图像。按照蛇形的行主顺序,灰度值依次为10,9,12,40,50,35,15,12,8,10,9,15,11,130,160和240。各像素所需的位数分别为4,4,4,6,6,6,4,4,4,4,4,4,4,8,8,8,按等长的条
21、件将像素分段,可以得到4个段10,9,12、40,50,35、15,12,8,10,9,15,11和130,160,240。因此,第一部分SegmentLength为2,2,6,2;第二部分BitsPerSegment的内容为3,5,3,7;第三部分Pixels包含了按蛇形行主顺序排列的16个灰度值,其中头三个各用4位存储,接下来三个各用6位,再接下来的七个各用4位,最后三个各用8位存储。因此存储单元中前30位存储了前六个像素:1010 1001 1100 111000 110010 100011这三个部分需要的存储空间分别为:第一部分SegmentLength需32位;BitsPerSegm
22、ent需12位;Pixels需82位,共需126位。而假如每个像素都用8位存储,则存储空间需816=128位,因而在本例图像中,节省了2位的空间。我们的问题不在这种方法能压缩多少而是由此引入问题:能否在此基础上,合并不同的段,从而减少所需的位数进而,找出最优合并假设在2)之后,产生了n个段。段标题(segment header)用于存储段的长度以及该段中每个像素所占用的位数。每个段标题需11位。现假设li和bi分别表达第i段的段长和该段每个像素的长度,则存储第i段像素所需要的空间为li*bi。在2)中所得的三个部分的总存储空间为11n+libi。可通过将某些相邻段合并的方式来减少空间消耗。如当
23、段i和i+1被合并时,合并后的段长应为li+li+1。此时每个像素的存储位数为maxbi,bi+1位。尽管这种技术增长了Pixels部分的空间消耗,但同时也减少了一个段标题的空间。例6-9假如将例6-8中的第1段和第2段合并,合并后,SegmentLength部分变为5,6,2,BitsPerSegment部分变为5,3,7。而Pixels部分的前36位存储的是合并后的第一段:001010 001001 001100 111000 110010 100011其余的像素(例6-8第3段)没有改变。由于减少了1个段标题, SegmentLength部分和BitsPerPixel部分的空间消耗共减少
24、了11位,而Pixels部分的空间增长6位,因此总共节约的空间为5位,空间总消耗为121位。我们希望能设计一种算法,使得在产生n个段之后,能对相邻段进行合并,以便产生一个具有最小空间需求的新的段集合。令sq为前q个段的最优合并所需要的空间。定义s0=0。考虑第i段(i0),假如在最优合并C中,第i段与第i-1,i-2,.,i-r+1段相合并,而不涉及第i-r段。合并C所需要的空间消耗等于:第1段到第i-r段所需空间+lsum(i-r+1,i)*bmax(i-r+1,i)+11其中lsum(a,b)=,bmax(a,b)=maxba,.,bb。假如在C中第1段到第i-r段的合并不是最优合并,那么
25、需要对合并进行修改,以使其具有更小的空间需求。因此还必须对段1到段i-r进行最优合并,也即保证最优原则得以维持。故C的空间消耗为:si=si-r+lsum(i-r+1,i)*bmax(i-r+1,i)+11r的值介于1到i之间,其中规定lsum不超过256(由于段长限制在256之内)。尽管我们不知道如何选择r,但我们知道,由于C具有最小的空间需求,因此在所有选择中,r必须产生最小的空间需求。因此可得递归式:假定kayi表达取得最小值时k的值,sn为n段的最优合并所需要的空间,因而一个最优合并可用kay的值构造出来。例6-10假定得到五个段,它们的长度为6,3,10,2,3,总计24个像素(24
26、BYTE),像素位数为1,2,3,2,1,要用公式计算sn,必须先求出sn-1,s0的值。s0为0,现计算s1:s1=s0+l1*b1 +11=17kay1=1s2由下式得出:s2=mins1+l2b2,s0+(l1+l2)*maxb1,b2+11=min17+6,0+9*2+11=29kay2=2以此类推,可得s1s5=17,29,67,73,82,kay1kay5=1,2,2,3,4。把s3写出来,可以看得更加清楚些:s3=mins2+l3b3 ,s1+(l2+l3)*maxb2,b3, s0+(l1+l2+l3)*maxb1,b2,b3+11由于s5=82,所以最优空间合并需82位的空间
27、。可由kay5导出本合并的方式,由于kay5=4,表达其最小值是在:s5 = s1 + lsum(2,5)*bmax(2,5)+11时取得的,可以得出合并的方案,即1,2-5两段。1)递归方法用递归式可以递归地算出si和kayi。程序6-3为递归式的计算代码。l,b,和kay是一维的全局整型数组,L是段长限制(256),header为段标题所需的空间(11)。调用S(n)返回sn的值且同时得出kay值。调用Traceback(kay,n)可得到最优合并。现给出程序6-3的复杂性:S的复杂性为t(n)=(2n)。Traceback的复杂性为(n)。程序6-3递归计算s,kay及最优合并int S
28、(int i)/返回S(i)并计算kayiif(i=0) return 0;/k=1时,根据公式(6-3)计算最小值int lsum=li,bmax=bi;int s=S(i-1)+lsum*bmax;kayi=1;/对其余的k计算最小值并求取最小值for(int k=2;k=i&lsum+li-k+1=L;k+)lsum+=li-k+1;if(bmaxt+lsum*bmax) s=t+lsum*bmax; kayi=k; return s+header;void Traceback(int n)/合并段if(n=0)return;Traceback(n-kayn);coutNew segme
29、nt begins at(n-kayn+1)0) return si;/已计算完/计算si/一方面根据公式(6-3)计算k=1时最小值int lsum=li,bmax=bi;si=S(i-1)+lsum*bmax;kayi=1;/对其余的k计算最小值并更新for(int k=2;k=i&lsum+li-k+1=L;k+)lsum+=li-k+1;if(bmaxt+lsum*bmax) si=t+lsum*bmax; kayi=k; si+=header;return si;可以证明程序6-4的时间复杂性为(n)。3)迭代方法倘若用式(6-3)依序计算s1,sn,便可得到一个复杂性为(n)的迭代
30、方法。在该方法中,在si计算之前,sj必须已计算好。该方法的代码见程序6-5,其中仍运用函数Traceback(见程序6-3)来获得最优合并。程序6-5迭代计算s和kayvoid Vbits(int n)/计算si和kayi int *s = new intn+1;int L=256,header=11;s0=0;/根据式(6-3)计算sifor(int i=1;i=n;i+)/k=1时,计算最小值int lsum=li,bmax=bi;si=si-1+lsum*bmax;kayi=1;/对其余的k计算最小值并更新for(int k=2;k=i&lsum+li-k+1=L;k+)lsum +=
31、 li-k+1;if(bmaxsi-k+lsum*bmax)si=si-k+lsum*bmax;kayi=k;si+=header;2.3矩阵乘法链mn矩阵A与np矩阵B相乘需花费(mnp)的时间。我们把mnp作为两个矩阵相乘所需时间的测量值。现假定要计算三个矩阵A、B和C的乘积,有两种方式计算此乘积。在第一种方式中,先用A乘以B得到矩阵D,然后D乘以C得到最终结果,这种乘法的顺序可写为(A*B)*C。第二种方式写为A*(B*C),道理同上。尽管这两种不同的计算顺序所得的结果相同,但时间消耗会有很大的差距。例6-12 一个较为极端的例子,假定A为1001矩阵,B为1100矩阵,C为1001矩阵
32、,则A*B的时间花费为10000,得到的结果D为100100矩阵,再与C相乘所需的时间花费为1000000,因此计算(A*B)*C的总时间为1010000。B*C的时间花费为10000,得到的中间矩阵为11矩阵,再与A相乘的时间消耗为100,因而计算A*(B*C)的时间花费竟只有10100!并且,计算(A*B)*C时,还需10000个单元来存储A*B,而A*(B*C)计算过程中,只需用1个单元来存储B*C。下面举一个得益于选择合适顺序计算A*B*C矩阵的实例:考虑两个3维图像的匹配。图像匹配问题的规定是,拟定一个图像需旋转、平移和缩放多少次才干逼近另一个图像。实现匹配的方法之一便是执行约100
33、次迭代计算,每次迭代需计算一个121向量T:T=A(x,y,z)*B(x,y,z)*C(x,y,z)其中A,B和C分别为123,33和31矩阵。(x,y,z)为矩阵中向量的坐标。设t表达计算A(x,y,z)*B(x,y,z)*C(x,y,z)的计算量。假定此图像含256256256个向量,在此条件中,这100个迭代所需的总计算量近似为100*2563*t1.7*109t。若三个矩阵是按由左向右的顺序相乘的,则t=12*3*3+12*3*1=144;但假如从右向左相乘,t=3*3*1+12*3*1=45。由左至右计算约需2.4*1011个操作,而由右至左计算大约只需7.5*1010个操作。假如使
34、用一个每秒可执行1000亿次操作的计算机,由左至右需2.4秒钟,而由右至左只需0.75秒钟。在计算矩阵运算A*B*C时,仅有两种乘法顺序(由左至右或由右至左),所以可以很容易算出每种顺序所需要的操作数,并选择操作数比较少的那种乘法顺序。但对于更多矩阵相乘来说,情况要复杂得多。如计算矩阵乘积M1M2Mq,其中Mi是一个riri+1矩阵(1iq)。不妨考虑q=4的情况,此时矩阵运算A*B*C*D可按以下方式(顺序)计算:A*(B*C)*D)A*(B*(C*D)(A*B)*(C*D)(A*(B*C)*D不难看出计算的方法数会随q以指数级增长。因此,对于很大的q来说,人脑已经无法考虑每一种计算顺序并选
35、择最优,这已是一个不切实际的问题。现在要介绍一种采用动态规划方法获得矩阵乘法顺序的最优策略。这种方法可将算法的时间消耗降为(q3)。用Mij表达链MiMj(ij)的乘积。设c(i,j)为用最优法计算Mij的消耗,kay(i,j)为用最优法计算Mij的最后一步MikMkj的k值。因此Mij的最优算法涉及如何用最优算法计算Mik和Mkj以及计算MikMkj。根据最优原理,可得到如下的动态规划递归式我们用矩阵的规模近似地当作运算的复杂性:以上求c的递归式可用递归或迭代的方法来求解。c(1,q)为用最优法计算矩阵链的消耗,kay(1,q)为最后一步的消耗。其余的乘积可由kay值来拟定。1)递归方法与求
36、解0/1背包及图像压缩问题同样,本递归方法也须避免反复计算c(i,j)和kay(i,j),否则算法的复杂性将会非常高。例6-13设q=5和r=(10,5,1,10,2,10),即:10*55*11*1010*22*10五个矩阵相乘。由动态规划的递归式得:式中待求的c中有四个c的s=0或1,因此用动态规划方法可立即求得它们的值:c(1,1)=c(5,5)=0;c(1,2)=50;c(4,5)=200。现计算C(2,5):c(2,5)=min c(2,2)+c(3,5)+50,c(2,3)+c(4,5)+500,c(2,4)+c(5,5)+100 其中c(2,2)=c(5,5)=0;c(2,3)=
37、50;c(4,5)=200。再用递归式计算c(3,5)及c(2,4):c(3,5)=min c(3,3)+c(4,5)+100,c(3,4)+c(5,5)+20 =min 0+200+100,20+0+20 = 40c(2,4)=min c(2,2)+c(3,4)+10,c(2,3)+c(4,4)+100 =min 0+20+10,50+10+20 = 30由以上计算还可得kay(3,5)=4,kay(2,4)=2。现在,计算c(2,5)所需的所有中间值都已求得,将它们代入式(6-5)得:c(2,5)=min 0+40+50,50+200+500,30+0+100 =90且kay(2,5)=2
38、。再用式(6-4)计算c(1,5),在此之前必须算出c(3,5)、c(1,3)和c(1,4)。同上述过程,亦可计算出它们的值分别为40、150和90,相应的kay值分别为4、2和2。代入式(6-4)得:c(1,5)=min 0+90+500,50+40+100,150+200+1000,90+0+200 =190且kay(1,5)=2此最优乘法算法的消耗为190,由kay(1,5)值可推出该算法的最后一步,kay(1,5)等于2,因此最后一步为M12M35,而M12和M35都是用最优法计算而来。由kay(1,2)=1知M12等于M11M22,同理由kay(3,5)=4得知M35由M34M55算
39、出。依此类推,M34由M33M44得出。因而此最优乘法算法的环节为:M11M22=M12M33M44=M34M34M55=M35M12M35=M15计算c(i,j)和kay(i,j)的递归代码见程序6-6。在函数C中,r为全局一维数组变量,kay是全局二维数组变量,函数C返回c(ij)之值且置kayab=kay(a,b)(对于任何a,b),其中c(a,b)在计算c(i,j)时皆已算出。函数Traceback运用函数C中已算出的kay值来推导出最优乘法算法的环节。设t(q)为函数C的复杂性,其中q=j-i+1(即Mij是q个矩阵运算的结果)。当q为1或2时,t(q)=d,其中d为一常数;而q2时
40、,其中e是一个常量。因此当q2时,t(q)2t(q-1)+e,所以t(q)=W (2q)。函数Traceback的复杂性为(q)。程序6-6递归计算c(i,j)和kay(i,j)int C(int i,int j)/返回c(i,j)且计算k(i,j)=kayijif(i=j) return 0;/一个矩阵的情形if(i=j-1)/两个矩阵的情形kayii+1=i;return ri*ri+1*ri+2;/多于两个矩阵的情形/设u为k=i时的最小值int u=C(i,i)+C(i+1,j)+ri*ri+1*rj+1;kayij=i;/计算其余的最小值并更新ufor(int k=i+1;kj;k+
41、)int t=C(i,k)+C(k+1,j)+ri*rk+1*rj+1;if(tu)/小于最小值的情形u=t;kayij=k;return u;void Traceback(int i,int j,int *kay)/输出计算Mij的最优方法if(i=j)return;Traceback(i,kayij,kay);Traceback(kayij+1,j,kay);coutMultiplyMi,kayij;coutandM(kayij+1),jend1;2)无反复计算的递归方法若避免再次计算前面已经计算过的c(及相应的kay),可将复杂性减少到(q3)。而为了避免反复计算,需用一个全局数组c存储c(i,j)值,该数组初始值为0。函数C的新代码见程序6-7:程序6-7无反复计算的c(i,j)计算方法int C(int i,int j)/返回c(i,j)并计算kay(i,j)=kayIj/避免反复计算/检查是否已计算过i
©2010-2025 宁波自信网络信息技术有限公司 版权所有
客服电话:4008-655-100 投诉/维权电话:4009-655-100