SPOJ-SERVICE 线性dp+维度压缩

Posted zsben991126

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了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);
    }
    
} 

 

以上是关于SPOJ-SERVICE 线性dp+维度压缩的主要内容,如果未能解决你的问题,请参考以下文章

线性代数的几何理解

hdu1024线性dp

一本通 高手训练 1781 死亡之树 状态压缩dp

出错记录关于DP中数组维度顺序

ML: 降维算法-概述

dp(未完成)