矩阵连乘问题 DP

题目大意:要你求矩阵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;
}


你可能感兴趣的:(矩阵连乘问题 DP)