完整word版动态规划算法设计与应用.docx
- 文档编号:23980432
- 上传时间:2023-05-23
- 格式:DOCX
- 页数:15
- 大小:38.84KB
完整word版动态规划算法设计与应用.docx
《完整word版动态规划算法设计与应用.docx》由会员分享,可在线阅读,更多相关《完整word版动态规划算法设计与应用.docx(15页珍藏版)》请在冰豆网上搜索。
完整word版动态规划算法设计与应用
实验报告
课程算法设计与分析实验实验名称动态规划算法设计与应用第1页
一、实验目的
1.加深对动态规划算法的基本原理的理解,掌握用动态规划方法求解最优化问题的方法步骤及应用;
2.用动态规划设计整数序列的最长递增子序列问题的算法,分析其复杂性,并实现;
3.用动态规划设计求凸多边形的三角剖分问题的算法,分析其复杂性,并实现。
4.选做题:
用动态规划设计求解0/1背包问题的算法,分析其复杂性,并实现。
二、实验内容
(一)最长递增子序列问题
1.问题描述
求一个由n个整数组成的整数序列的最长递增子序列。
一个整数序列的递增子序列可以是序列中非连续的数按照原序列顺序排列而成的。
最长递增子序列是其递增子序列中长度最长的。
2.具体要求(若在ACM平台上提交程序,必须按此要求)――平台上1700题
输入:
输入的第一行是一个正整数n,表示测试例个数。
接下来几行是n个测试例的数据,每个测试例的数据由两行组成,其中第一行为一个正整数k(k<=500),表示整数序列的长度,第二行给出整数序列,整数之间用一个空格隔开。
(设给出的每个整数序列的最长递增子序列都是唯一的。
)
输出:
对于每个测试例输出两行,第一行为最长递增子序列的长度,第二行为最长递增子序列,整数之间用一个空格隔开。
两个测试例的输出数据之间用一个空行隔开,最后一个测试例后无空行。
3.测试数据
输入:
3
5
31423
6
139526
20
1271335102412491653683823113147
输出:
3
123
4
1356
10
1235101216233147
4.设计与实现的提示
(1)寻找最优子结构、写出递归方程是问题的关键。
(2)以Ai为末元素的最长递增子序列(记为S(i)),等于以使S(j),(j=1~i),最大的那个Aj为末元素的递增子序列最末再加上Ai;如果这样的元素不存在,那么Ai自身构成一个长度为1的以Ai为末元素的递增子序列。
(3)最优解的信息在此是以Ai为末元素的最长递增子序列的前驱元素,应当记录下来。
5.扩展内容
本题可以采用多种方法求解,可以尝试用不同思路求解。
(二)凸多边形的三角剖分
1.问题描述
设P是一个有n个顶点的凸多边形,P中的弦是P中连接两个非相邻顶点的线段。
用P中的(n-3)条弦将P剖分成(n-2)个三角形(如下图所示)。
使得(n-3)条弦的长度之和最小的三角形剖分称为最优三角剖分。
2.具体要求(若在ACM平台上提交程序,必须按此要求)――平台上1701题
输入:
输入的第一行是一个正整数m,表示测试例个数,接下来几行是m个测试例的数据,每个测试例的数据由两行组成,第一行含一个正整数n(n<=500),表示凸多边形的顶点个数;第二行含2n个实数x1,y1,x2,y2,…xn,yn,按顺时针方向依次给出n个顶点的坐标值(xi,yi)i=1,2,…,n,整数之间用一个空格隔开。
输出:
对于每个测试例输出一行,含一个实数(精确到小数点后三位),表示最优三角剖分的n-3条弦的长度之和。
两个测试例的输出数据之间用一个空行隔开,最后一个测试例后无空行。
3.测试数据
输入:
2
6
1221.520.51000.501.5
9
72312204631074370842317534524192992871378355168385513011131
输出:
5.606
4928.722
4.设计与实现的提示
(1)凸(n+1)边形的最优三角剖分是该凸多边形的所有三角剖分中弦长最短的那个剖分。
(2)本题与课本7.3节中的矩阵链相乘问题非常相似。
寻找最优子结构、写出递归方程是问题的关键,注意递归方程的参数的设置。
(3)将凸(n+1)边形三角剖分时,注意弦长不要重复计算。
5.扩展内容
本题只要求计算出最优值。
如果要求最优解,在求最优值过程中,要记录下改最优值对应的三角剖分的信息。
(三)选做题――0/1背包问题
1.问题描述
设有一个容量为C的背包,n个物品的集合U={u1,u2,…,un},物品uj的体积和价值分别为sj和vj,C,sj,vj都是正整数。
在U中选择物品装入背包,使得装入背包的物品总价值最大。
设每种物品或完全装入或完全不装入背包。
2.具体要求
输入:
输入的第一行是一个正整数m,表示测试例个数,接下来几行是m个测试例的数据。
每个测试例的数据由三行组成,第一行含两个正整数n和C,其中,n(n<=100)表示给定的是n个物品的集合U={u1,u2,…,un},C(C<=5000)表示背包的容量;第二行含n个整数s1,s2,…sn,表示n个物品的体积;第三行含n个整数v1,v2,…vn,表示n个物品的价值,整数之间用一个空格隔开。
输出:
对于每个测试例输出2行数据,其中,第一行含一个整数,表示装入背包物品的最大总价值;第2行含n个整数x1,x2,…xn,表示u1,u2,…,un这n个物品是否放入背包。
其中xi={1,0},(i=1,2,…,n)。
如果xi=1,表示物品ui放入背包;如果xi=0,表示物品ui不放入背包。
每个xi之间用一个空格隔开,两个测试例的输出数据之间用一个空行隔开,最后一个测试例后无空行。
3.测试数据
输入:
2
520
713643
57323
8100
713452516754832
6520107322213
输出:
13
1011
48
10100010
4.设计与实现的提示
(1)寻找最优子结构、写出递归方程仍是问题求解的关键。
(2)对于物品是否装入背包,可以有多种表示方式,这里仅采用0,1的方式,即xi=1表示物品ui装入背包;xi=0表示物品ui不装入背包。
(3)由于在求解最优值时,并非所有的子问题都需要解,因此可以采用自顶向下的递归方法。
(4)在求解最优值的过程中,注意最优解的相应信息的表示和记录。
5.扩展内容
(1)可以练习实现课本第7章的练习7.27可复选的背包问题,并与本题进行比较。
(2)可以练习实现课本第7章的练习7.26。
要求:
取不同K值,求0/1背包问题的近似最优值,观察分析不同的K值对最优值的精确性的影响。
(3)可以将本题与一般背包问题(一个物品可以部分装入背包)相比较,可以问题的复杂性和求解方法上有何不同。
三、实验环境
硬件:
WindowsXP计算机、鼠标、键盘、显示器
开发环境:
MicrosoftVisualC++6.0
四、实验步骤
(描述实验步骤及中间的结果或现象。
在实验中做了什么事情,怎么做的,发生的现象和中间结果)
1、点击开始菜单中的程序-MicrosoftVisualC++6.0
点击菜单栏中的文件—新建—文件—C++SourceFile,在文件名(N)中写入“实验四
(1).cpp”,再点击确定.
2、编写程序如下:
#include
#defineMAX50
/*--------------------------------冒泡排序-----------------------*/
voidbubble(intr[],intn,intB[])//使用冒泡排序,将原数组按升序排列
{
inti,j,temp;
for(i=0;i { for(j=0;j if(r[j]>r[j+1]) { temp=r[j]; r[j]=r[j+1]; r[j+1]=temp; } } for(i=0;i { B[i]=r[i]; } } /*------------------------求序列A和B的最长公共子序列C-------------------*/ voidLcss(intH[MAX][MAX],intn,intA[MAX],intL,intB[MAX]) { inti=n,j=n,k=L,C[MAX]; while(i>0&&j>0) { if(H[i][j]==0) { C[k--]=A[i]; i=i-1; j=j-1; } else if(H[i][j]==1) i=i-1; else j=j-1; C[0]=B[0]; } for(i=0;i { printf("%4d",C[i]); } } /*-------求A和B的最长公共子序列长度L[n,m]和存储最长公共子序列的有关信息的数组H[1..n,1..m]-----*/ voidLcs(intA[MAX],intB[MAX],intn,intnum) { inti,j,length; intL[MAX][MAX],H[MAX][MAX]; for(i=0;i L[i][0]=0; for(j=0;j L[0][j]=0; for(i=1;i<=n;i++) { for(j=1;j<=n;j++) { if(A[i]==B[j]) { L[i][j]=L[i-1][j-1]+1; H[i][j]=0; } else if(L[i-1][j]>=L[i][j-1]) { L[i][j]=L[i-1][j]; H[i][j]=1; } else { L[i][j]=L[i][j-1]; H[i][j]=2; } } } printf("%d",L[n][n]); length=L[n][n]; printf("\n"); printf("第%d个子序列元素: ",num+1); Lcss(H,n,A,length,B); } /*---------------------主函数----------------------*/ voidmain() { intm,i,j,A[MAX],B[MAX],C[MAX],D[MAX]; printf("测试例的个数: \n"); scanf("%d",&m); for(i=0;i { printf("第%d个的长度及元素: \n",i+1); scanf("%d",&C[i]); for(j=0;j { scanf("%d",&A[j]); D[j]=A[j]; } bubble(A,C[i],B);//调用函数,使用冒泡排序将A按升序排列 printf("\n"); printf("第%d个子序列长度: ",i+1); Lcs(D,B,C[i],i);//调用函数,求输入的元素与已排序的元素的最长公共子序列 printf("\n\n"); } } 3、点击开始菜单中的程序-MicrosoftVisualC++6.0 点击菜单栏中的文件—新建—文件—C++SourceFile,在文件名(N)中写入“实验四 (2).cpp”,再点击确定. 4、编写程序如下: #include #include floatdistance[100][100]; #defineMAX100 /*-------------求凸边形的各个顶点与其它顶点的距离------------------*/ voiddis(intcount,floatx[MAX],floaty[MAX]) { inti,j; for(i=0;i { for(j=0;j { distance[i][j]=0; } } for(i=0;i { for(j=0;j { distance[i][j]=sqrt((x[i]-x[j])*(x[i]-x[j])+(y[i]-y[j])*(y[i]-y[j])); } } } /*-------------------------最优三角剖分的弦长之和-----------------------*/ floattriangle(intzero,intnum) { intk,m=0; floats=40000,a[MAX];//设置一个较大的数s与弦长之和比较 if(num-zero==2)//若凸边形为三角形时,不存在剖分 { return0; } else { for(k=zero+1;k { if(k==zero+1)//假设第一个三角形的边的两个点是原点和点zero+1,那么剩下的还是一个凸多边形 { a[m]=triangle(k,num)+distance[zero+1][num]; } elseif(k==num-1)//假设第一个三角形的边的两个点是原点和点num,那么剩下的还是一个凸多边形 { a[m]=triangle(zero,k)+distance[zero][num-1]; } else { a[m]=triangle(k,num)+distance[k][num]+triangle(zero,k)+distance[zero][k]; } m++; } for(m=0;m { if(s>a[m]) { s=a[m]; } } returns; } } /*-----------------------主函数----------------------*/ voidmain() { intm,i,j,A[MAX]; floatx[MAX],y[MAX],length[MAX]; printf("输入测例个数: \n"); scanf("%d",&m); for(i=0;i { printf("输入第%d测例数的顶点数为: \n\n",i+1); scanf("%d",&A[i]);
- 配套讲稿:
如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
- 特殊限制:
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。
- 关 键 词:
- 完整 word 动态 规划 算法 设计 应用
![提示](https://static.bdocx.com/images/bang_tan.gif)