UVA - 1601 The Morning after Halloween (双向BFS&单向BFS)
题目:
w*h(w,h≤16)网格上有n(n≤3)个小写字母(代表鬼)。要求把它们分别移动到对应的大写字母里。每步可以有多个鬼同时移动(均为往上下左右4个方向之一移动),但每步结束之后任何两个鬼不能占用同一个位置,也不能在一步之内交换位置。输入保证所有空格连通,所有障碍格也连通,且任何一个2*2子网格中至少有一个障碍格。输出最少的步数。
分析:
1.又碰到了“状态”这个很玄乎的词了。以当前三个鬼的位置为状态,那就有256^3(16*16=256)个,然后三个鬼在每一个状态中的移动有5^3个,这时间和空间肯定受不了啊。
2.紫书上说,可以将空格提取出来,然后对其BFS就可以了……读完有些懵逼提取?那还不是遍历?看完代码才明白可以对每个空格子进行编号,然后存一下每个格子可以走向的格子的编号就提取出来了!!其中还对每一个状态进行了编号,这也太妙了!!还是太菜了……之后BFS就可以了。
单向BFS代码:
#include <bits/stdc++.h>
#define inf 0x3f3f3f3f
#define MAX 1000000009
#define FRE() freopen("in.txt","r",stdin)
#define FRO() freopen("out.txt","w",stdout)
using namespace std;
typedef long long ll;
const int maxn = ;
const int dx[] = {-,,,,};
const int dy[] = {,,-,,};
inline int ID(int a,int b,int c) {
return (a<<)|(b<<)|c;
}
int s[],t[];//保存初末状态
int deg[maxn];//某个格子有多少个与之相连的格子
int G[maxn][];//保存某个格子可以去向哪些格子
inline bool conflict(int a,int b,int a2,int b2) {
return a2==b2||(a2==b&&b2==a);//如果两个鬼移动到同一个位置或位置互换则冲突
}
int d[maxn][maxn][maxn];//走到某个状态经过的步数 int bfs(){
queue<int>q;
memset(d,-,sizeof(d));
q.push(ID(s[],s[],s[]));
d[s[]][s[]][s[]] = ;
while(!q.empty()){
int u=q.front();q.pop();
int a=(u>>)&0xff,b=(u>>)&0xff,c=u&0xff;//获取该状态下,三个鬼的位置 if(a==t[]&&b==t[]&&c==t[])//如果已经找到了,就返回结果
return d[a][b][c];
for(int i=; i<deg[a]; i++){//在一步中三个鬼到底该怎么走,那就三重循环枚举所有的可能情况就可以了
int a2=G[a][i];
for(int j=; j<deg[b]; j++){
int b2=G[b][j];
if(conflict(a,b,a2,b2))continue;
for(int k=; k<deg[c]; k++){
int c2=G[c][k];
if(conflict(a,c,a2,c2)) continue;
if(conflict(b,c,b2,c2)) continue;
if(d[a2][b2][c2]!=-) continue;
d[a2][b2][c2] = d[a][b][c]+;//记录距离
q.push(ID(a2,b2,c2));
}
}
}
}
return -;
} int main() {
//FRE();
int w,h,n;
while(scanf("%d%d%d\n",&w,&h,&n)== && n){
char maze[][];
for(int i=; i<h; i++){
fgets(maze[i],,stdin);
}
int cnt,x[maxn],y[maxn],id[maxn][maxn];
cnt=;
for(int i=; i<h; i++){//提取图中所有可以走的格子
for(int j=; j<w; j++){
if(maze[i][j]!='#'){
x[cnt]=i,y[cnt]=j,id[i][j]=cnt;//记录格子并编号
if(islower(maze[i][j])) s[maze[i][j]-'a'] = cnt;//记录初始位置
else if(isupper(maze[i][j]))t[maze[i][j]-'A'] = cnt;//记录末位置
cnt++;
}
}
}
for(int i=; i<cnt; i++){//遍历提取的格子
deg[i] = ;
for(int dir=; dir<; dir++){//记录可以从该格子走到的格子
int nx=x[i]+dx[dir],ny=y[i]+dy[dir];
if(maze[nx][ny]!='#')
G[i][deg[i]++] = id[nx][ny];
}
}
//如果起始格子不够3个就补全,让他们的起始位置和末位置重合就可以了
if(n==){
deg[cnt] = ; G[cnt][]=cnt;s[]=t[] = cnt++;
}
if(n==){
deg[cnt] = ; G[cnt][]=cnt;s[]=t[] = cnt++;
}
printf("%d\n",bfs());
}
return ;
}
这个题目也可以用双向BFS来做,做完这个题,对双向BFS有了进一步的理解。那这里的正向就是遍历小写字母,反向就是遍历大写字母。当小写字母和大写字母的位置重合之后,就是得到的答案。
双向BFS代码:
#include <bits/stdc++.h>
#define inf 0x3f3f3f3f
#define MAX 1000000009
#define FRE() freopen("in.txt","r",stdin)
#define FRO() freopen("out.txt","w",stdout)
using namespace std;
typedef long long ll;
const int maxn = ;
const int dx[] = {-, , , , };
const int dy[] = {, , -, , };
inline int ID(int a, int b, int c) {
return (a << ) | (b << ) | c;
}
int s[], t[]; //保存初末状态
int deg[maxn];//某个格子有多少个与之相连的格子
int G[maxn][];//保存某个格子可以去向哪些格子
inline bool conflict(int a, int b, int a2, int b2) {
return a2 == b2 || (a2 == b && b2 == a); //如果两个鬼移动到同一个位置或位置互换则冲突
}
int d[maxn][maxn][maxn], vis[maxn][maxn][maxn]; //走到某个状态经过的步数 int bfs() {
queue<int> que,que_rev;
d[s[]][s[]][s[]] = ;
d[t[]][t[]][t[]] = ; que.push(ID(s[],s[],s[]));
que_rev.push(ID(t[],t[],t[])); vis[s[]][s[]][s[]] = ;
vis[t[]][t[]][t[]] = ; while(!que.empty() && !que_rev.empty()){
int sza = que.size(),szb=que_rev.size();
while(sza--){
int u=que.front(); que.pop();
int a=(u>>)&0xff,b=(u>>)&0xff,c=(u&0xff);
for(int i=; i<deg[a]; i++){
int a2 = G[a][i];
for(int j=; j<deg[b]; j++){
int b2 = G[b][j];
if(conflict(a,b,a2,b2)) continue;
for(int k=; k<deg[c]; k++){
int c2 = G[c][k];
if(conflict(a,c,a2,c2)) continue;
if(conflict(b,c,b2,c2)) continue; if(vis[a2][b2][c2]==){
vis[a2][b2][c2] = ;
d[a2][b2][c2] = d[a][b][c]+;
que.push(ID(a2,b2,c2));
}else if(vis[a2][b2][c2]==){
return d[a2][b2][c2]+d[a][b][c]+;
}
}
}
}
}
while(szb--){
int u=que_rev.front(); que_rev.pop();
int a=(u>>)&0xff, b=(u>>)&0xff,c=(u&0xff);
for(int i=; i<deg[a]; i++){
int a2 = G[a][i];
for(int j=; j<deg[b]; j++){
int b2=G[b][j];
if(conflict(a,b,a2,b2))continue;
for(int k=; k<deg[c]; k++){
int c2=G[c][k];
if(conflict(a,c,a2,c2))continue;
if(conflict(b,c,b2,c2))continue; if(vis[a2][b2][c2]==){
vis[a2][b2][c2]=;
d[a2][b2][c2] = d[a][b][c]+;
que_rev.push(ID(a2,b2,c2));
}else if(vis[a2][b2][c2]==){
return d[a2][b2][c2]+d[a][b][c]+;
}
}
}
}
}
}
return -;
} int main() {
//FRE();
int w, h, n;
while(scanf("%d%d%d\n", &w, &h, &n) == && n) {
char maze[][];
memset(d, -, sizeof(d));
memset(vis, , sizeof(vis));
int cnt, x[maxn], y[maxn], id[maxn][maxn];
cnt = ; for(int i = ; i < h; i++) {
fgets(maze[i], , stdin);
} for(int i = ; i < h; i++) { //提取图中所有可以走的格子
for(int j = ; j < w; j++) {
if(maze[i][j] != '#') {
x[cnt] = i, y[cnt] = j, id[i][j] = cnt; //记录格子并编号
if(islower(maze[i][j])) {
s[maze[i][j] - 'a'] = cnt; //记录初始位置
} else if(isupper(maze[i][j])) {
t[maze[i][j] - 'A'] = cnt; //记录末位置
}
cnt++;
}
}
}
for(int i = ; i < cnt; i++) { //遍历提取的格子
deg[i] = ;
for(int dir = ; dir < ; dir++) { //记录可以从该格子走到的格子
int nx = x[i] + dx[dir], ny = y[i] + dy[dir];
if(maze[nx][ny] != '#')
{ G[i][deg[i]++] = id[nx][ny]; }
}
}
//如果起始格子不够3个就补全,让他们的起始位置和末位置重合就可以了
if(n == ) {
deg[cnt] = ;
G[cnt][] = cnt;
s[] = t[] = cnt++;
}
if(n == ) {
deg[cnt] = ;
G[cnt][] = cnt;
s[] = t[] = cnt++;
}
printf("%d\n", bfs());
}
return ;
}
UVA - 1601 The Morning after Halloween (双向BFS&单向BFS)的更多相关文章
- UVA - 1601 The Morning after Halloween (BFS/双向BFS/A*)
题目链接 挺有意思但是代码巨恶心的一道最短路搜索题. 因为图中的结点太多,应当首先考虑把隐式图转化成显式图,即对地图中可以相互连通的点之间连边,建立一个新图(由于每步不需要每个鬼都移动,所以每个点需要 ...
- UVA 1601 The Morning after Halloween
题意: 给出一个最大为16×16的迷宫图和至多3个ghost的起始位置和目标位置,求最少经过几轮移动可以使三个ghost都到达目标位置.每轮移动中,每个ghost可以走一步,也可以原地不动,需要注意的 ...
- UVa 1601 || POJ 3523 The Morning after Halloween (BFS || 双向BFS && 降维 && 状压)
题意 :w*h(w,h≤16)网格上有n(n≤3)个小写字母(代表鬼).要求把它们分别移动到对应的大写字母里.每步可以有多个鬼同时移动(均为往上下左右4个方向之一移动),但每步结束之后任何两个鬼不能占 ...
- <<操作,&0xff以及|的巧妙运用(以POJ3523---The Morning after Halloween(UVa 1601)为例)
<<表示左移,如a<<1表示将a的二进制左移一位,加一个0,&0xff表示取最后8个字节,如a&0xff表示取a表示的二进制中最后8个数字组成一个新的二进制数, ...
- [uva P1601] The Morning after Halloween
[uva P1601] The Morning after Halloween 题目链接 非常经典的一道题目,lrj的书上也有(貌似是紫书?). 其实这题看起来就比较麻烦.. 首先要保证小鬼不能相遇, ...
- POJ1915Knight Moves(单向BFS + 双向BFS)
题目链接 单向bfs就是水题 #include <iostream> #include <cstring> #include <cstdio> #include & ...
- HDU1195 双向BFS(或BFS)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1195 , 双向BFS或者直接BFS也可以过. 其实这道题只是单向BFS就可以过的,但是为了练算法,所以 ...
- POJ 2251 Dungeon Master --- 三维BFS(用BFS求最短路)
POJ 2251 题目大意: 给出一三维空间的地牢,要求求出由字符'S'到字符'E'的最短路径,移动方向可以是上,下,左,右,前,后,六个方向,每移动一次就耗费一分钟,要求输出最快的走出时间.不同L层 ...
- POJ1475 Pushing Boxes(BFS套BFS)
描述 Imagine you are standing inside a two-dimensional maze composed of square cells which may or may ...
随机推荐
- 双层for 循环
说说 图片 集合 list(set([ii for i in img_url_l for ii in i[0].replace('[','').replace(']','').replace(&quo ...
- YTU 2541: 汽水瓶
2541: 汽水瓶 时间限制: 1 Sec 内存限制: 128 MB 提交: 40 解决: 27 题目描述 有这样一道智力题:"某商店规定:三个空汽水瓶可以换一瓶汽水.小张手上有十个空汽 ...
- [译]NUnit--Installation(三)
Installation NUnit安装程序默认安装文件路径为C:\Program Files\NUnit 2.6.2.根据用户选择安装的选项,安装文件有三个子文件夹:bin.doc.samples. ...
- [HNOI2005]星际贸易
https://www.zybuluo.com/ysner/note/1309789 题面 要素太多,还是自己看吧 解析 如果要求贸易额最大,就相当于: 有\(n\)个物品(星球),每个物品价值为\( ...
- XHTML与HTML区别
1.一下规则适用于XHTML,但并不适用于HTML <html>.<head>.<body>都是必需的标签 必须设置<html>标签的xmlns属性,且 ...
- Oracle.ManagedDataAccess.dll
C#使用Oracle.ManagedDataAccess.dll System.Data.OracleClient程序集没有多大的不同,基本上拿以前使用System.Data.OracleClient ...
- 2017西安网络赛B_Coin
样例输入 2 2 1 1 3 1 2 样例输出 500000004 555555560 思路: n重伯努利实验概率分布题. 设q=1-p,p为事件概率. Y为出现偶数次的概率. 所以 Y=1/2*( ...
- Node“getTextContent() is undefined for the type Node”处理办法
最近一个项目在MyEclipse导入后总是报getTextContent() is undefined for the type Node错误. 经过查找原来是因为Node类为JDK中自带的(org. ...
- NLog简单配置与使用
对项目添加NLog 安装完成后,在项目里面会自动引入该引入的dll,并且会添加如下两个文件 NLog的配置主要是在这个config文件里.当然也可以将这个文件里面的nlog节点复制到项目配置文件App ...
- virtualwrapper使用
1.创建: mkvirtualenv 虚拟环境目录 2.查看虚拟环境 workon lsvirtualenv 3.进入虚拟环境 workon 虚拟环境目录 4.退出虚拟环境 deactivate 5. ...