51nod1572 宝岛地图

勇敢的水手们到达了一个小岛,在这个小岛上,曾经有海盗在这里埋下了一些宝藏。然而,我们的船快抛锚了,与此同时,船长发现藏宝图的一角被老鼠咬掉了一块。

 

藏宝图可以用一个n×m大小的矩形表示。矩形中的每一小块表示小岛中的一小块陆地(方块的边长为1米)。有一些方块表示的是海,这些块人是不能通过的。除了海不能走,其它的小方块都是可以行走的。在可行走区域里有一些小方块表示一些已知的地点。

 

另外,在地图上有k条指令。每条指令的格式表示如下:

“向y方向走n米”。

 

这里的方向有四种:“北”,“南”,“东”,“西”。如果你正确的跟着这些指令行走,并且完整的执行完所有指令,你就可以找到宝藏所在的地点。

 

但是,很不幸,由于地图中好多地方都缺失了,船长也不知道从哪些地方开始走。但是船长依然清楚地记得一些已知的地点。另外,船长也知道所有可行走区域。

 

现在船长想知道从哪些已知地点出发,按照指令,可能找到宝藏所在地。

Input
单组测试数据
第一行包含两整数n和m(3≤n,m≤1000)。
接下来的n行每行有m个字符,表示整个地图。
“#”代表海。在地图矩形中,矩形的四周一圈一定是海。
“.”代表可行走区域,未知地点。大写字母“A”到“Z”表示可行走区域,已知地点。
所有大写字母不一定都被用到。每个字母在地图中最多出现一次。所有已知地点用不同的大写字母表示。

接下来一行有一个整数k(1≤k≤10^5),接下来有k行。
每行表示一条指令。
指令格式为“dir len”,“dir”表示朝哪个方向走,“len”表示走几步。
“dir”有四种取值“N”,“S”,“E”,“W”,对应题目中的“北”,“南”,“东”,“西”
在地图中,北是在顶部,南是在底部,西是在左边,东是在右边。“len”是一个整数,范围在[1,1000]。
Output
共一行,按字典序升序打印出所有可以完整执行地图中指令的已知区域的字母,如果没有满足要求的已知区域,则打印“no solution”(没有引号)。
Input示例
输入样例1
6 10
##########
#K#..#####
#.#..##.##
#..L.#...#
###D###A.#
##########
4
N 2
S 1
E 1
W 2
Output示例
输出样例1
AD
本来我以为要建图跑KMP的,后来发现预处理+模拟就可以了。预处理出每个点向4个方向的最大长度,模拟。
#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
using namespace std;
char str[1005][1005];
int f[1005][1005][4];
int n,m,num,k,dir[100005],len[100005];
struct note
{
    char ch;
    int x,y;
    friend bool operator<(note a,note b)
    {
        return a.ch<b.ch;
    }
}que[1000005];
/*
    north:0 east:1 south:2 west:3 
*/
bool check(int x,int y)
{
    
    for(int i=1;i<=k;i++)
    {
        //cout<<x<<' '<<y<<endl;
        if (f[x][y][dir[i]]<=len[i]) 
        {
            return false;
            //cout<<"\n\n\n";
        }
        if(dir[i]==0)x-=len[i];else
            if(dir[i]==1)y+=len[i];else
                if(dir[i]==2)x+=len[i];else
                     y-=len[i];
    }
    //cout<<x<<' '<<y<<endl;;
    return true;
}
void prepare()
{
    scanf("%d %d",&n,&m);
    for(int i=1;i<=n;i++)
    {
        scanf("%s",str[i]+1);
        int len=strlen(str[i]+1); 
        for(int j=1;j<=len;j++)
        {
            if(str[i][j]!='#')
            {
                f[i][j][0]=f[i-1][j][0]+1;
                f[i][j][3]=f[i][j-1][3]+1;
            }
            if(str[i][j]>='A'&&str[i][j]<='Z')
            {
                que[++num]=(note){str[i][j],i,j};
            }
        }
    }
    for(int i=n;i>=1;--i)
    {
        int len=strlen(str[i]+1); 
        for(int j=len;j>=1;j--)
        {
            if(str[i][j]!='#')
            {
                f[i][j][2]=f[i+1][j][2]+1;
                f[i][j][1]=f[i][j+1][1]+1;
            }
        }
    }
}
void read_info()
{
    char ch1[5];
    scanf("%d",&k);
    for(int i=1;i<=k;i++)
    {
        scanf("%s %d",ch1,&len[i]);
        if(ch1[0]=='N') dir[i]=0;
            else if(ch1[0]=='E') dir[i]=1;
                else if(ch1[0]=='S') dir[i]=2;
                    else dir[i]=3;
    }
}
void debug()
{
    for(int pp=0;pp<=3;pp++)
    {
        for(int i=1;i<=n;i++)
        {
            for(int j=1;j<=m;j++)
            cout<<f[i][j][pp]<<' ';
            cout<<endl;
        }
        cout<<"\n\n\n"<<endl;
    }
}
int main()
{
    prepare();
    //debug();
    read_info();
    sort(que+1,que+1+num);
    bool oo=0;
    for(int i=1;i<=num;i++)
    if(check(que[i].x,que[i].y))
    {
        cout<<que[i].ch;
        oo=1;
    }
    if(!oo) cout<<"no solution"<<endl;
 } 

 

posted @ 2017-08-11 09:37  dancer16  阅读(143)  评论(0编辑  收藏  举报
描述