题意:
给出一个迷宫从(1,1)到(n,m),迷宫的每个格子有一个分数,走了这个格子就可以得到这个格子的分数,如何得到最大的分数和。
题解:
和以往的dp不同,这个可以走回头路,相当于绕来绕去,这样不能用普通的dp做,插头dp解决之,模板改下就过了2a。
#include<iostream>
#include<math.h>
#include<stdio.h>
#include<algorithm>
#include<string.h>
#include<vector>
#include<map>
using namespace std;
typedef __int64 lld;
#define oo 0x3f3f3f3f
#define OO 0x3f3f3f3f3f3f3f3f
#define HASH 10007
#define STATE 1000010
#define MAXD 15
int N,M,ex,ey;
int code[MAXD];
int maze[MAXD][MAXD];
int ch[MAXD];
int scores[MAXD][MAXD];
struct HASHMAP
{
int head[HASH],next[STATE],sizes;
int dp[STATE];
lld state[STATE];
void init()
{
sizes=0;
memset(head,-1,sizeof head);
}
void push(lld st,int ans)
{
int h=st%HASH;
for(int i=head[h];i!=-1;i=next[i])
{
if(st==state[i])
{
dp[i]=max(dp[i],ans);
return ;
}
}
dp[sizes]=ans;
state[sizes]=st;
next[sizes]=head[h];
head[h]=sizes++;
}
}hm[2];
void decode(int code[],int m,lld st)
{
for(int i=m;i>=0;i--)
{
code[i]=st&7;
st>>=3;
}
}
lld encode(int code[],int m)///最小表示法
{
lld st=0;
int cnt=0;
memset(ch,-1,sizeof ch);
ch[0]=0;
for(int i=0;i<=m;i++)
{
if(ch[code[i]]==-1) ch[code[i]]=++cnt;
code[i]=ch[code[i]];
st<<=3;
st|=code[i];
}
return st;
}
void shift(int code[],int m)///换行 移位
{
for(int i=m;i>0;i--)
code[i]=code[i-1];
code[0]=0;
}
void dpblank(int i,int j,int cur)
{
int left,up;
for(int k=0;k<hm[cur].sizes;k++)
{
decode(code,M,hm[cur].state[k]);
left=code[j-1];
up=code[j];
///开头
if(i==1&&j==1&&left==0&&up==0)
{
///两个决策,是往右走还是往下走
if(maze[i][j+1])//往右走
{
code[j-1]=0;
code[j]=13;
hm[cur^1].push(encode(code,M),hm[cur].dp[k]+scores[i][j]);
}
if(maze[i+1][j])//往下走
{
code[j-1]=13;
code[j]=0;
if(j==M)shift(code,M);
hm[cur^1].push(encode(code,M),hm[cur].dp[k]+scores[i][j]);
}
continue;
}
///结尾必须有插头,切只有一个否则到达不了
if(i==N&&j==M&&((left&&!up)||(!left&&up)))
{
code[j-1]=code[j]=0;
if(j==M)shift(code,M);
hm[cur^1].push(encode(code,M),hm[cur].dp[k]+scores[i][j]);
continue;
}
if(left&&up)///11 -> 00 有上插头和左插头,这种情况下相当于合并两个连通分量
{
if(left!=up)///不存在环才进行dp
{
code[j-1]=code[j]=0;
for(int t=0;t<=M;t++)
if(code[t]==up)
code[t]=left;
if(j==M)shift(code,M);
hm[cur^1].push(encode(code,M),hm[cur].dp[k]+scores[i][j]);
}
}
else if(left||up)///01 || 10 上插头和左插头恰好有一个,这种情况相当于延续原来的连通分量
{
int temp;
if(left) temp=left;
else temp=up;
if(maze[i][j+1])
{
code[j-1]=0;
code[j]=temp;
hm[cur^1].push(encode(code,M),hm[cur].dp[k]+scores[i][j]);
}
if(maze[i+1][j])
{
code[j-1]=temp;
code[j]=0;
if(j==M)shift(code,M);///切记不可忘记,换行要shift
hm[cur^1].push(encode(code,M),hm[cur].dp[k]+scores[i][j]);
}
}
else///没有上插头和左插头,有下插头和右插头,相当于构成一个新的连通块
{
///这里有两个决策,第一个很正场必须有,建立性的插头,第二决策不建立插头(不走这条路呗)
if(maze[i][j+1]&&maze[i+1][j])
{
code[j]=code[j-1]=13;
hm[cur^1].push(encode(code,M),hm[cur].dp[k]+scores[i][j]);
}
code[j]=code[j-1]=0;
if(j==M)shift(code,M);
hm[cur^1].push(encode(code,M),hm[cur].dp[k]);///因为不走这条路所以不用加分数
}
}
}
void init()
{
memset(maze,0,sizeof maze);
for(int i=1;i<=N;i++)
{
for(int j=1;j<=M;j++)
{
maze[i][j]=1;
scanf("%d",&scores[i][j]);
}
}
}
void solve()
{
int cur=0;
lld ans=0;
hm[cur].init();
hm[cur].push(0,0);
for(int i=1;i<=N;i++)
for(int j=1;j<=M;j++)
{
hm[cur^1].init();
dpblank(i,j,cur);
cur^=1;
}
for(int i=0;i<hm[cur].sizes;i++)
ans+=hm[cur].dp[i];
printf("%I64d\n",ans);
}
int main()
{
int cas=1;
while(scanf("%d %d",&N,&M)!=EOF)
{
init();
printf("Case %d: ",cas++);
if(N==1&&M==1)
{
printf("%d\n",scores[1][1]);
continue;
}
solve();
}
return 0;
}
/**
2
2 2
OO
O*
4 4
***O
XO**
**O*
XX**
*/