SPOJ-SERVICE 线性dp+维度压缩

还是线性dp,有点感觉了,另外这个问题也可以用滚动数组

/*
依然是先按照阶段i划分,
dp[i][j][k]表示完成第i个请求时,两个员工分别在j位置和k位置的费用(还有一个员工一定在位置p)

dp[i][j][k]=min(dp[i][j][k],dp[i-1][j][k]+cost[pre][p]);处理完上一个请求的员工处理下一个请求,其余两个不懂
dp[i][j][pre]=min(dp[i][j][pre],dp[i-1][j][k]+cost[k][p]);处理完上一个请求的员工不动,余下两个中的一个处理下一个请求
dp[i][k][pre]=min(dp[i][k][pre],dp[i-1][j][k]+cost[j][p]);处理完上一个请求的员工不动,余下两个中的一个处理下一个请求
*/ 
#include<iostream>
#include<cstring>
#include<cstdio>
#include<algorithm>
using namespace std;

int n,l,c[201][201],list[1001],dp[2][201][201];

int main(){
    int t;
    scanf("%d",&t);
    while(t--){
        scanf("%d%d",&l,&n);
        for(int i=1;i<=l;i++)
            for(int j=1;j<=l;j++)
                scanf("%d",&c[i][j]);
        for(int i=1;i<=n;i++) scanf("%d",&list[i]);
        memset(dp,0x7f,sizeof dp);
        int las=1,now=0,ans=0x7fffffff,pre,p;
        dp[0][1][2]=c[3][list[1]]; 
        dp[0][1][3]=c[2][list[1]];
        dp[0][2][3]=c[1][list[1]];
        
        for(int i=2;i<=n;i++){//依次处理每个订单 
            las^=1,now^=1;pre=list[i-1],p=list[i];//滚动数组翻转一下
            memset(dp[now],0x7f,sizeof dp[now]);//把原来的数据清空
            for(int i=1;i<=l;i++)
                for(int j=1;j<=l;j++){
                    if(i==j||i==pre||j==pre) continue;
                    dp[now][i][j]=dp[now][j][i]=min(dp[now][i][j],dp[las][i][j]+c[pre][p]);//第一种情况
                    dp[now][i][pre]=dp[now][pre][i]=min(dp[now][i][pre],dp[las][i][j]+c[j][p]);
                    dp[now][j][pre]=dp[now][pre][j]=min(dp[now][j][pre],dp[las][i][j]+c[i][p]); 
                }
        }    
        
        for(int i=1;i<=l;i++) 
            for(int j=1;j<=l;j++)
                if(i!=j && i!=list[n] && j!=list[n])ans=min(ans,dp[now][i][j]);
        printf("%d
",ans);
    }
    
} 
原文地址:https://www.cnblogs.com/zsben991126/p/10214546.html