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 now;//现在钥匙状态
	int x,y;
	int step;
	friend bool operator<(Node aa,Node bb) {
		return aa.step > bb.step;
	}
};

void init() {
	memset(vis,false,sizeof(vis));
}

bool input() {
    while(cin>>n>>m>>t) {
		for(int i=0;i<n;i++) {
			scanf("%s",mp[i]);
			for(int j=0;j<m;j++) {
				if(mp[i][j] == '@')sx = i,sy = j;
				if(mp[i][j] == '^')ex = i,ey = j;
			}
		}
        return false;
    }
    return true;
} 

int ans;

void bfs(int x,int y) {
	queue<Node> q;
	Node ss,ee;
	ss.x = x,ss.y = y,ss.step = 0,ss.now = 0;
	q.push(ss);
	vis[ss.x][ss.y][0] = true;
	while(!q.empty()) {
		ss = q.front();
		q.pop();
		if(ss.step >= t)continue;
		if(ss.x == ex && ss.y == ey) {
			ans = min(ans,ss.step);
			break;
		}
		for(int i=0;i<4;i++) {
			int dx = ss.x + dir[i][0];
			int dy = ss.y + dir[i][1];
			if(dx < 0 || dx >= n || dy < 0 || dy >= m)continue;
			if(mp[dx][dy] == '*')continue;
			if(vis[dx][dy][ss.now])continue;
			ee.now = ss.now;
			if(mp[dx][dy] >= 'A' && mp[dx][dy] <= 'J') {/*遇到门*/
				int tmp = mp[dx][dy] - 'A';
				if((ss.now&(1<<tmp)) == 0)continue;
			}
			if(mp[dx][dy] >= 'a' && mp[dx][dy] <= 'z') {/*遇到钥匙*/
				int tmp = mp[dx][dy] - 'a';
				ee.now |= (1<<tmp);
			}
			ee.x = dx,ee.y = dy,ee.step = ss.step + 1;
			vis[dx][dy][ss.now] = true;
			q.push(ee);
		}
	}
}

void cal() {
	ans = inf;
	bfs(sx,sy);
	if(ans == inf || ans >= t)puts("-1");
	else cout<<ans<<endl;
}

void output() {

}

int main() {
    while(true) {
        init();
        if(input())return 0;
        cal();
        output();
    }
    return 0;
}
时间: 2024-08-10 23:07:09

HDU1429 胜利大逃亡(续) BFS +简单状压的相关文章

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

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

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

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

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

Problem Description Ignatius再次被魔王抓走了(搞不懂他咋这么讨魔王喜欢)-- 这次魔王汲取了上次的教训,把Ignatius关在一个n*m的地牢里,并在地牢的某些地方安装了带锁的门,钥匙藏在地牢另外的某些地方.刚开始Ignatius被关在(sx,sy)的位置,离开地牢的门在(ex,ey)的位置.Ignatius每分钟只能从一个坐标走到相邻四个坐标中的其中一个.魔王每t分钟回地牢视察一次,若发现Ignatius不在原位置便把他拎回去.经过若干次的尝试,Ignatius已画

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&

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.胜利大逃亡(续)(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+状压

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

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

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1429 思路分析:题目要求找出最短的逃亡路径,但是与一般的问题不同,该问题增加了门与钥匙约束条件: 考虑一般的搜索问题的解答思路: 搜索算法即在解空间中搜索满足要求的答案,可以看做一棵不断生长的状态树,状态之间不断扩展出新的状态,直到找出所需要的状态,即答案: <1>定义状态:对于该问题,由于存在门与锁的约束条件,所以状态应该包括3个元素,即人所在的坐标 x 和 y 以及含有锁的种类: <2&