题目大意:要你求矩阵A1*A2*A3*A4*A5*A6…………*An的最优解法,就是求乘的次序,因为一个i*j的矩阵*一个j*k的矩阵时间复杂度为i*j*k,通过适当的改变运算顺序,可以使原来的问题规模变小。这里给你若干矩阵的行列数,求计算顺序。
解题思路:一道典型的dp,算法老师在课上讲的,然后留的作业,不难。令dp[i][j],为计算从第i到j这段区间的矩阵所需的最小次数,其可以由dp[i+1][j]+相应3个行列数的乘积更新来,同理也可以从右边更新。本题要求的是计算次序,而不是最优解的大小,所以在算出每个区间的最优解的时候,都要把他划分的位置记录下来以便以后查阅,我记录到了goback[][]。之后就是反向推导出计算顺序,我用了个Back()记录下了相关信息到ans[][]里,最后output,书上的样例能过,但是不知道有没有bug
/*测试数据 6 30 35 35 15 15 5 5 10 10 20 20 25 */ #include <iostream> #include <stdio.h> #include <math.h> #include <stdlib.h> #include <string> #include <string.h> #include <algorithm> #include <vector> #include <queue> #include <set> #include <map> #include <stack> using namespace std; typedef long long LL; const int INF=0x7fffffff; const int MAX_N=1009; int N,C; int A[MAX_N][2]; int dp[MAX_N][MAX_N]; int goback[MAX_N][MAX_N]; int ans[MAX_N][3]; int solve(int x,int y){ if(dp[x][y]!=-1)return dp[x][y]; if(x==y)return dp[x][y]=0; if(x+1==y)return dp[x][y]=A[x][0]*A[x][1]*A[y][1]; int ans=INF,t; for(int i=x;i<=y-1;i++){ t=ans; ans=min(ans,solve(x,i)+solve(i+1,y)+A[x][0]*A[i][1]*A[y][1]); if(t!=ans){ goback[x][y]=i; } } return dp[x][y]=ans; } void Back(int x,int y){ if(y-x==0)return; if(y-x==1){ ans[C][1]=x; ans[C][2]=y; C++; return; } ans[C][1]=x; ans[C][2]=y; C++; Back(x,goback[x][y]); Back(goback[x][y]+1,y); } void output(){ int ct=C-1; for(int i=1;i<=C-1;i++){ printf("第%d步:计算",i); printf("%d~%d\n",ans[ct][1],ans[ct][2]); ct--; } } int main(){ printf("输入矩阵的个数:"); while(scanf("%d",&N)!=EOF){ C=1; memset(dp,-1,sizeof(dp)); printf("下面输入矩阵的维数,共%d行\n",N); for(int i=1;i<=N;i++){ scanf("%d%d",&A[i][0],&A[i][1]); } cout<<"乘积为"<<solve(1,N)<<endl; Back(1,N); output(); printf("\n输入矩阵的个数:"); } return 0; }