hdu1429胜利大逃亡(续) (状态压缩+BFS)

Problem Description

Ignatius再次被魔王抓走了(搞不懂他咋这么讨魔王喜欢)……

这次魔王汲取了上次的教训,把Ignatius关在一个n*m的地牢里,并在地牢的某些地方安装了带锁的门,钥匙藏在地牢另外的某些地方。刚开始Ignatius被关在(sx,sy)的位置,离开地牢的门在(ex,ey)的位置。Ignatius每分钟只能从一个坐标走到相邻四个坐标中的其中一个。魔王每t分钟回地牢视察一次,若发现Ignatius不在原位置便把他拎回去。经过若干次的尝试,Ignatius已画出整个地牢的地图。现在请你帮他计算能否再次成功逃亡。只要在魔王下次视察之前走到出口就算离开地牢,如果魔王回来的时候刚好走到出口或还未到出口都算逃亡失败。

Input

每组测试数据的第一行有三个整数n,m,t(2<=n,m<=20,t>0)。接下来的n行m列为地牢的地图,其中包括:

. 代表路

* 代表墙

@ 代表Ignatius的起始位置

^ 代表地牢的出口

A-J 代表带锁的门,对应的钥匙分别为a-j

a-j 代表钥匙,对应的门分别为A-J

每组测试数据之间有一个空行。

Output

针对每组测试数据,如果可以成功逃亡,请输出需要多少分钟才能离开,如果不能则输出-1。

Sample Input

4 5 17
@A.B.
a*.*.
*..*^
c..b*

4 5 16
@A.B.
a*.*.
*..*^
c..b*

Sample Output

16
-1

每个位置都有不同钥匙的状态经过。
#include<stdio.h>
#include<iostream>
#include<string.h>
#include<queue>
using namespace std;

typedef struct nnn
{
    int t,key,x,y;
}NODE;
int Key[25][25][1<<10],T,n,m;
char map[25][25];
int abs(int x)
{
    return x>=0?x:-x;
}
int bfs(int sx,int sy)
{
    queue<NODE>q;
    NODE now,pre;
    int dir[4][2]={0,1,0,-1,1,0,-1,0};
    int tx,ty;
    now.key=0; now.t=0; now.x=sx; now.y=sy;
    Key[sy][sx][0]=1;
    q.push(now);
    while(!q.empty())
    {
        pre=q.front(); q.pop();
        for(int i=0; i<4; i++)
        {
            now=pre;
            ty=pre.y+dir[i][0]; tx=pre.x+dir[i][1];
            if(ty>=0&&ty<n&&tx>=0&&tx<m&&pre.t<T-1&&map[ty][tx]!=‘*‘)
            {
                now.t++; now.y=ty; now.x=tx;
                if(map[ty][tx]==‘^‘) return now.t;
                if(map[ty][tx]>=‘a‘&&map[ty][tx]<=‘j‘){
                    now.key|=(1<<(map[ty][tx]-‘a‘));
                    if(Key[ty][tx][now.key]==0)
                    Key[ty][tx][now.key]=1,q.push(now);
                }
                else if(map[ty][tx]>=‘A‘&&map[ty][tx]<=‘J‘)
                {
                    if(Key[ty][tx][now.key]==0&&(pre.key&(1<<(map[ty][tx]-‘A‘))))
                    Key[ty][tx][now.key]=1, q.push(now);
                }
                else if(Key[ty][tx][now.key]==0){
                    Key[ty][tx][now.key]=1; q.push(now);
                }
            }
        }
    }
    return -1;
}
int main()
{
    int sx,sy,ex,ey;
    while(scanf("%d%d%d",&n,&m,&T)>0)
    {
        for(int i=0; i<n; i++)
        scanf("%s",map[i]);
        for(int y=0 ; y<n; y++)
        for(int x=0 ; x<m; x++)
        if(map[y][x]==‘@‘)
        sy=y,sx=x;
        else if(map[y][x]==‘^‘)
        ey=y,ex=x;
        if(abs(sy-ey)+abs(sx-ex)>=T)
        {
            printf("-1\n"); continue;
        }
        memset(Key,0,sizeof(Key));
        printf("%d\n",bfs(sx,sy));
    }
}

hdu1429胜利大逃亡(续) (状态压缩+BFS)

时间: 2024-10-30 14:23:57

hdu1429胜利大逃亡(续) (状态压缩+BFS)的相关文章

hdu1429胜利大逃亡(续)(状态压缩+bfs)

题目链接: 啊哈哈,点我点我 题意及思路 最开始我以为跟普通的bfs一样,所以直接写了一个朴素的bfs,一跑,前两组数据对了,但是第三组不对,一看,走过的还可以走啊,所以不能标记,结果我的bfs乱改,最后 毫无疑问改成了死循环.所以看题解... 思路:因为有10中不同的钥匙,每种都有两种状态,所以结合计算机是二进制保存的特点,刚好把这10把钥匙当成每一个为,要要1<<10个位保存所有的状态,然后就是模拟捡起钥匙,捡起钥匙就是说明这个位上的数字变成1这个状态,所以自然而然想到了位运算,只要|一下

hdu-1429 胜利大逃亡(续)

http://acm.hdu.edu.cn/showproblem.php?pid=1429 第一次接触搜索+状态压缩    看了大神的题解  勉强把题目弄懂了. 用二进制来表示手头的钥匙有哪些,100表示有第三把钥匙,111表示有第三.二.一把,搜索下一点时,如果该点为钥匙点,则可采用|运算来 模拟拾取,显然0001 | 1000 = 1001,同理,当为相应的门时采用&运算来模拟开启,例如1101 & 0001 = 0001(即可以打开'A'门) #include<cstdio&

HDU1429 胜利大逃亡(续)(BFS+状态压缩)

题目链接:点击打开链接 题意:迷宫中,一个起点,一个终点,迷宫中有墙,有门,门的钥匙也在迷宫中某处,只有拿到钥匙才能打开门,问能不能再T步(不含)之内逃出迷宫. 题解:在朴素BFS上增加了钥匙的状态,只有有钥匙才能打开门,总共有不超过10吧钥匙,所以用一个int的整数的二进制即可存储钥匙的状态.碰到门先判断状态,碰到钥匙更新状态. 代码: #include<iostream> #include<cstdio> #include<cstring> #include<

HDU1429胜利大逃亡(续)BFS+状态压缩

这题的算是BFS中应用状压的一个模板题吧,没啥难度,用key来存储已获得的钥匙,状压一下就可以了 不过我写的过程中,犯了好多SB错误,导致调试了好久才A,本来仔细可以1A的说 #include <cstdio> #include <cstring> #include <cmath> #include <algorithm> #include <climits> #include <string> #include <iostre

HDU1429 胜利大逃亡(续) BFS +简单状压

把手中持有的钥匙状态状压一下即可,然后vis访问标记的时候,开个三维,多一维即为当前持有钥匙状态,这样就能祛除重复标记困难走点的问题,跟网络赛那题很像,网络赛的更难点,这个简单点 int n,m,t; int sx,sy,ex,ey; char mp[20 + 55][20 + 55]; bool vis[20 + 5][20 + 5][(1<<10) + 5]; int dir[4][2] = {{1,0},{0,1},{-1,0},{0,-1}}; struct Node { int no

胜利大逃亡(续)(状态压缩bfs)

胜利大逃亡(续) Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total Submission(s): 7357    Accepted Submission(s): 2552 Problem Description Ignatius再次被魔王抓走了(搞不懂他咋这么讨魔王喜欢)……这次魔王汲取了上次的教训,把Ignatius关在一个n*m的地牢里,并在地牢的某些地方安装了带

hdoj 1429 胜利大逃亡(续) 【BFS+状态压缩】

题目:hdoj 1429 胜利大逃亡(续) 相同题目: 题意:中文的,自己看 分析:题目是求最少的逃亡时间,确定用BFS 这个题目的难点在于有几个锁对于几把钥匙,唯一的对应关系,不能用直接的标记法,因为一个图可能需要搜索多次. 仔细分析的话会发现,图的搜索次数是和钥匙的出现次数相关,那么我们可以用二进制的0 和 1 来表示第几把钥匙出现过没有,所以我们可以用状态压缩来标记那个钥匙出现过,然后用三维标记,第三维表示出现几个钥匙了的情况下图的点的搜索情况.其他就和简单的一样. AC代码: #incl

HDU 1429 胜利大逃亡(续)(bfs+状态压缩,很经典)

传送门: http://acm.hdu.edu.cn/showproblem.php?pid=1429 胜利大逃亡(续) Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Submission(s): 10648    Accepted Submission(s): 3860 Problem Description Ignatius再次被魔王抓走了(搞不懂他咋这么讨魔王

HDU 1429胜利大逃亡(续) (bfs+状态压缩)

胜利大逃亡(续) Time Limit: 4000/2000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) Total Submission(s): 6469 Accepted Submission(s): 2243 Problem Description Ignatius再次被魔王抓走了(搞不懂他咋这么讨魔王喜欢)-- 这次魔王汲取了上次的教训,把Ignatius关在一个n*m的地牢里,并在地牢的某些地方安装了带锁的门,钥