hdu-1429 胜利大逃亡(续)

http://acm.hdu.edu.cn/showproblem.php?pid=1429

第一次接触搜索+状态压缩    看了大神的题解  勉强把题目弄懂了。

用二进制来表示手头的钥匙有哪些,100表示有第三把钥匙,111表示有第三、二、一把,搜索下一点时,如果该点为钥匙点,则可采用|运算来

模拟拾取,显然0001 | 1000 = 1001,同理,当为相应的门时采用&运算来模拟开启,例如1101 & 0001 = 0001(即可以打开‘A‘门)

#include<cstdio>
#include<cstring>
#include<queue>
using namespace std;
int n,m,t,ans;
char map[25][25];
int vis[25][25][1030];//标记数组
int dir[4][2]={-1,0,1,0,0,1,0,-1}; //四个方向
struct point
{
    int x,y,step,key;
};
point s;
int check(int x,int y)//检查
{
    if(s.x>=0&&s.x<n&&s.y>=0&&s.y<m)
        return 1;
    return 0;
}
void bfs()
{
    point tp;
    queue<point>q;
    vis[s.x][s.y][s.key]=1;
    for(q.push(s);!q.empty();q.pop())
    {
        tp=q.front();
        for(int k=0;k<4;k++)
        {
            s=tp; //注意这里要先把tp赋值给s,即先把上次搜索到的保存下来,再去扩展。
            s.x=tp.x+dir[k][0];
            s.y=tp.y+dir[k][1];
            if(check(s.x,s.y)&&map[s.x][s.y]!='*'&&!vis[s.x][s.y][s.key])
            {
                if(map[s.x][s.y]=='.'||map[s.x][s.y]=='@')
                {
                    vis[s.x][s.y][s.key]=1;
                    s.step++;
                    q.push(s);
                }
                if(map[s.x][s.y]=='^')
                {
                    if(s.step+1<t)
                        ans=s.step+1;
                    return;
                }
                if(map[s.x][s.y]>='A'&&map[s.x][s.y]<='J')
                {//a<<2表示将a是二进制左移2位(相当于*4),这里用来表示钥匙eg:A-A=0;1<<0,就是0000 0001就是第一把钥匙了。
                //同理当g[s.x][s.y]='B'时,就是1<<2,就是0000 0010,表示第二把钥匙;
                    int key=1<<(map[s.x][s.y]-'A');//按位于运算,当有对应的钥匙时,等式成立;才能打开门
                    if(s.key&key)
                    {
                        vis[s.x][s.y][s.key]=1;
                        s.step+=1;
                        q.push(s);
                    }
                }
                if(map[s.x][s.y]>='a'&&map[s.x][s.y]<='j')
                {  //按位或运算,吸收钥匙;
                    int key=1<<(map[s.x][s.y]-'a');
                    if(!vis[s.x][s.y][s.key|key])
                    {
                        vis[s.x][s.y][s.key|key]=1;
                        s.step+=1;
                        s.key=s.key|key;
                        q.push(s);
                    }
                }
            }
        }
    }

}

int main()
{
    while(scanf("%d%d%d",&n,&m,&t)!=EOF)
    {
        memset(vis,0,sizeof(vis));
        ans=-1;
        for(int i=0;i<n;i++)
        {
            scanf("%s",map[i]);
            for(int j=0;j<m;j++)
            {
                if(map[i][j]=='@')
                {
                    s.x=i;
                    s.y=j;
                    s.key=0;
                    s.step=0;
                }
            }
        }
        bfs();
        printf("%d\n",ans);
    }
    return 0;
}

hdu-1429 胜利大逃亡(续),布布扣,bubuko.com

时间: 2024-12-17 08:00:42

hdu-1429 胜利大逃亡(续)的相关文章

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): 6270    Accepted Submission(s): 2177 Problem Description Ignatius再次被魔王抓走了(搞不懂他咋这么讨魔王喜欢)-- 这次魔王汲取了上次的教训,把Ignatius关在一个n*m的地牢里,并在地牢的某些地方安装了

hdu 1429 胜利大逃亡(续)(bfs+位压缩)

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

hdu 1429 胜利大逃亡(续)【广度优先搜索+状态压缩】

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

hdu 1429 胜利大逃亡(续) 搜索+状态压缩,,不错的题。

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

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的地牢里,并在地牢的某些地方安装了带锁的门,钥

hdu.1429.胜利大逃亡(续)(bfs + 0101011110)

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

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

http://acm.hdu.edu.cn/showproblem.php?pid=1429 终于开始能够做状态压缩的题了,虽然这只是状态压缩里面一道很简单的题. 状态压缩就是用二进制的思想来表示状态. 总共有10种钥匙,那么开一个(1<<10 的数组) 那么每次遇到一把钥匙我就用当前状态 |  钥匙转化为2进制的数值,然后遇到门的时候判断是否有对应的钥匙,只要用当前状态 & 门转化为2进制的值就可以.初始为0时,state=0,表示10个状态位都是0.那么每次遇到钥匙就改变相应的状态

HDU 1429 胜利大逃亡(续) BFS+状压

状压一下然后随意写,注意如果你被魔王抓了一次钥匙就全丢了哦,这样第二个样例就可以解释为什么是-1而不是20了 #include <cstdio> #include <cstring> #include <iostream> #include <map> #include <set> #include <vector> #include <string> #include <queue> #include <

hdu 1429 胜利大逃亡(续)(BFS+位压缩)

#include <stdio.h> #include <string.h> #include <iostream> #include <algorithm> #include <vector> #include <queue> #include <stack> #include <set> #include <map> #include <string> #include <ma