测试地址:Prison Break
题目大意:一个n×m的网格中,一些格子是空地,一些格子是开关(一开始状态为开),一些格子是能量池,还有一些格子不能走。要求从一个给定的格子出发,关闭所有的开关,每走一格会耗费一个单位能量,走到能量池可以使用它来回复能量到满电池容量,也可以走过不使用,能量池只能使用一次,问要关闭所有开关所需的最小的电池容量是多少。出发时是满能量的。
做法:本题需要用到状态压缩DP+BFS+二分答案。
首先,注意到电池容量大于某一个阈值时,都是可以完成任务的,反之就都不能完成,答案具有单调性,可以二分。接下来就是判定对于一个给定的电池容量mid,存不存在一种方案使得能开完所有开关。
由于能量池和开关都是只能使用一次(使用多次没有意义),而且它们的数量很少(15个),所以可以套用经典的状压DP模型——TSP(旅行商)问题来解决,只需要BFS预处理出这些点两两之间需要消耗的能量,然后设f(i,j)为节点使用过的状态为i时,最后走到点j的最大剩余能量,然后就可以列出方程计算了,写的时候要注意一些细节,就可以完成这个问题了。
以下是本人代码:

#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <iostream>
#include <algorithm>
using namespace std;
int n,m,map[21][21],tot,p[21][2],t[21],final,start;
int g[21][21],dis[21][21],f[40010][21];
int q[410][2],head,tail;
bool vis[21][21];
char s[21];
struct forsort {int val,bit;} st[40010];

bool cmp(forsort a,forsort b)
{
    return a.bit<b.bit;
}

void bfs(int s)
{
    head=1,tail=1;
    memset(vis,0,sizeof(vis));
    q[1][0]=p[s][0],q[1][1]=p[s][1];
    dis[q[1][0]][q[1][1]]=0;
    vis[q[1][0]][q[1][1]]=1;
    while(head<=tail)
    {
        int x=q[head][0],y=q[head][1];
        if (x>1&&!map[x-1][y]&&!vis[x-1][y]) dis[x-1][y]=dis[x][y]+1,vis[x-1][y]=1,q[++tail][0]=x-1,q[tail][1]=y;
        if (x<n&&!map[x+1][y]&&!vis[x+1][y]) dis[x+1][y]=dis[x][y]+1,vis[x+1][y]=1,q[++tail][0]=x+1,q[tail][1]=y;
        if (y>1&&!map[x][y-1]&&!vis[x][y-1]) dis[x][y-1]=dis[x][y]+1,vis[x][y-1]=1,q[++tail][0]=x,q[tail][1]=y-1;
        if (y<m&&!map[x][y+1]&&!vis[x][y+1]) dis[x][y+1]=dis[x][y]+1,vis[x][y+1]=1,q[++tail][0]=x,q[tail][1]=y+1;
        head++;
    }
    for(int i=1;i<=tot;i++)
    {
        if (vis[p[i][0]][p[i][1]]) g[s][i]=dis[p[i][0]][p[i][1]];
        else g[s][i]=-1;
    }
}

bool check(int mid)
{
    for(int i=0;i<(1<<tot);i++)
        for(int j=1;j<=tot;j++)
            f[i][j]=-1;
    f[1<<(start-1)][start]=mid;
    for(int i=2;i<=(1<<tot);i++)
    {
        int v=st[i].val,w[21]={0};
        int x=v,j=1;
        while(x)
        {
            if (x&1) w[++w[0]]=j;
            x>>=1;j++;
        }
        for(j=1;j<=w[0];j++)
        {
            for(int k=1;k<=w[0];k++)
                if (k!=j)
                {
                    if (t[w[k]]==1&&g[w[k]][w[j]]!=-1&&f[v-(1<<(w[j]-1))][w[k]]!=-1)
                        f[v][w[j]]=max(f[v][w[j]],mid-g[w[k]][w[j]]);
                    else if (g[w[k]][w[j]]!=-1&&f[v-(1<<(w[j]-1))][w[k]]!=-1)
                        f[v][w[j]]=max(f[v][w[j]],f[v-(1<<(w[j]-1))][w[k]]-g[w[k]][w[j]]);
                }
        }
    }
    for(int i=0;i<(1<<tot);i++)
        if ((i&final)==final)
        {
            for(int j=1;j<=tot;j++)
                if (f[i][j]>=0) return 1;
        }
    return 0;
}

int main()
{
    while(scanf("%d%d",&n,&m)&&n&&m)
    {
        final=0;
        tot=0;
        memset(map,0,sizeof(map));
        for(int i=1;i<=n;i++)
        {
            scanf("%s",s);
            for(int j=1;j<=m;j++)
            {
                if (s[j-1]=='D') map[i][j]=1;
                if (s[j-1]=='F') p[++tot][0]=i,p[tot][1]=j,t[tot]=0,start=tot;
                if (s[j-1]=='G') p[++tot][0]=i,p[tot][1]=j,t[tot]=1;
                if (s[j-1]=='Y') p[++tot][0]=i,p[tot][1]=j,t[tot]=2,final+=1<<(tot-1);
            }
        }

        for(int i=1;i<=tot;i++)
            bfs(i);
        for(int i=0;i<(1<<tot);i++)
        {
            st[i+1].val=i;
            st[i+1].bit=0;
            int x=i;
            while(x)
            {
                if (x&1) st[i+1].bit++;
                x>>=1;
            }
        }
        sort(st+1,st+(1<<tot)+1,cmp);

        int l=0,r=n*m+1;
        while(l<r)
        {
            int mid=(l+r)>>1;
            if (check(mid)) r=mid;
            else l=mid+1;
        }
        if (l!=n*m+1) printf("%d\n",l);
        else printf("-1\n");
    }

    return 0;
}