BZOJ 4761 Cow Navigation

Posted on 2017-02-01 10:17  ziliuziliu  阅读(201)  评论(0编辑  收藏  举报

一开始以为瞎jb写个IDA*就过了。。然而并不是这样。(naive)

我的做法是dp[x1][y1][x2][y2][d1][d2],表示两头奶牛所在的位置和面朝的方向,然后直接spfa搞定。

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<queue>
#define maxn 25
using namespace std;
int n,map[maxn][maxn],dis[maxn][maxn][maxn][maxn][5][5];
char s[maxn];
int dx[]={0,-1,0,1,0},dy[]={0,0,1,0,-1};
bool vis[maxn][maxn][maxn][maxn][5][5];
struct status
{
    int x1,x2,y1,y2,d1,d2;
    status (int x1,int y1,int x2,int y2,int d1,int d2):x1(x1),y1(y1),x2(x2),y2(y2),d1(d1),d2(d2) {}
    status () {}
};
queue <status> q;
bool judge(int x,int y)
{
    return ((x>=1) && (x<=n) && (y>=1) && (y<=n));
}
void spfa()
{
    q.push(status(n,1,n,1,1,2));
    while (!q.empty())
    {
        status head=q.front();q.pop();
        int nx1,nx2,ny1,ny2;
        nx1=head.x1+dx[head.d1];nx2=head.x2+dx[head.d2];ny1=head.y1+dy[head.d1];ny2=head.y2+dy[head.d2];
        int t2=dis[head.x1][head.y1][head.x2][head.y2][head.d1][head.d2];
        if ((!judge(nx1,ny1)) || (!map[nx1][ny1])) nx1=head.x1,ny1=head.y1;
        if ((!judge(nx2,ny2)) || (!map[nx2][ny2])) nx2=head.x2,ny2=head.y2;
        if ((head.x1==1) && (head.y1==n)) nx1=1,ny1=n;
        if ((head.x2==1) && (head.y2==n)) nx2=1,ny2=n;
        int &t1=dis[nx1][ny1][nx2][ny2][head.d1][head.d2];
        if (t1>t2+1)
        {
            t1=t2+1;
            bool &t3=vis[nx1][ny1][nx2][ny2][head.d1][head.d2];
            if (!t3)
            {
                t3=true;
                q.push(status(nx1,ny1,nx2,ny2,head.d1,head.d2));
            }
        }
        int dd1,dd2;
        dd1=head.d1-1;dd2=head.d2-1;if (!dd1) dd1=4;if (!dd2) dd2=4;
        int &t5=dis[head.x1][head.y1][head.x2][head.y2][dd1][dd2];
        if (t5>t2+1)
        {
            t5=t2+1;
            bool &t3=vis[head.x1][head.y1][head.x2][head.y2][dd1][dd2];
            if (!t3)
            {
                t3=true;
                q.push(status(head.x1,head.y1,head.x2,head.y2,dd1,dd2));
            }
        }
        dd1=head.d1+1;dd2=head.d2+1;if (dd1==5) dd1=1;if (dd2==5) dd2=1;
        int &t4=dis[head.x1][head.y1][head.x2][head.y2][dd1][dd2];
        if (t4>t2+1)
        {
            t4=t2+1;
            bool &t3=vis[head.x1][head.y1][head.x2][head.y2][dd1][dd2];
            if (!t3)
            {
                t3=true;
                q.push(status(head.x1,head.y1,head.x2,head.y2,dd1,dd2));
            }
        }
    }
}
int main()
{
    scanf("%d",&n);memset(dis,0x3f,sizeof(dis));
    for (int i=1;i<=n;i++)
    {
        scanf("%s",s);
        for (int j=0;j<n;j++)
        {
            if (s[j]=='E')
                map[i][j+1]=1;
        }
    }
    dis[n][1][n][1][1][2]=0;
    spfa();
    int mn=0x3f3f3f3f;
    for (int i=1;i<=4;i++)
        for (int j=1;j<=4;j++)
            mn=min(mn,dis[1][n][1][n][i][j]);
    printf("%d\n",mn);
    return 0;
}