uva 1601 poj 3523 Morning after holloween 万圣节后的早晨 (经典搜索,双向bfs+预处理优化+状态压缩位运算)
这题数据大容易TLE
优化:预处理, 可以先枚举出5^3的状态然后判断合不合法,但是由于题目说了有很多墙壁,实际上没有那么多要转移的状态那么可以把底图抽出来,然后3个ghost在上面跑到时候就不必判断了,减少了两次无用的枚举。
减少代码的方法:1.结点没有3个时增加冗余点,2.把位置坐标二元组编号成一个数,这点在预处理时可以顺便完成(坐标范围0~15),3.把三个ghost的位置状态压缩成一个数字,方便push,但是注意重时不能直接用Hash掉的值来判断vis,因为Hash以后数字范围很大。
这题为什么不适合Astar。因为Astar的估计用Manhattan距离需要知道坐标,因此不适合把二元组hash成数字,而且不能用一个数来压缩状态了,因为要加上估价值,代码很长,容易写错。
学习点:
1.减少代码量的方法,减少分类,提取不同的问题的相同部分
2.建图的方法。
poj不支持#include<bits/stdc++.h>
//Rey
#include<bits/stdc++.h>
using namespace std;
const int maxw = ;
const int maxn = ;//14*14*3/4+2 149 int s[];
int t[];
int w,h,n;
char maze[maxw][maxw+]; int deg[maxn],G[maxn][]; inline bool conflict(int a1,int b1,int a2,int b2){
return a1 == a2 || (a1 == b2 && a2 == b1);
}
int vis1[maxn][maxn][maxn];
int vis2[maxn][maxn][maxn];
typedef vector<int> VINT;
VINT v1;
VINT v2;
VINT v3;
typedef VINT * PV; inline int Hash(int a,int b,int c) {return a<<|b<<|c;}
int dBfs()
{
v1.clear();v2.clear();v3.clear();
memset(vis1,-,sizeof(vis1));
memset(vis2,-,sizeof(vis2));
PV q1 = &v1,q2 = &v2,nxt = &v3;
int (*d1)[maxn][maxn] = vis1, (*d2)[maxn][maxn] = vis2;
d1[s[]][s[]][s[]] = ;
d2[t[]][t[]][t[]] = ;
q1->push_back(Hash(s[],s[],s[]));
q2->push_back(Hash(t[],t[],t[]));
while(q1->size()&&q2->size()){
if(q1->size()>q2->size()) swap(q1,q2),swap(d1,d2);
for(int ii = ,sz = q1->size(); ii < sz; ii++){
int u = (*q1)[ii];
int a = u>>, b = (u>>)&0xff, c = u&0xff;
for(int i = ; i < deg[a]; i++){
int a1 = G[a][i];
for(int j = ; j < deg[b]; j++){
int b1 = G[b][j];
if(conflict(a1,a,b1,b)) continue;
for(int k = ; k < deg[c]; k++){
int c1 = G[c][k];
if(conflict(c1,c,a1,a)||conflict(c1,c,b1,b)||~d1[a1][b1][c1]) continue;
d1[a1][b1][c1] = d1[a][b][c]+;
if(~d2[a1][b1][c1]) { return d1[a1][b1][c1]+d2[a1][b1][c1]; }
nxt->push_back(Hash(a1,b1,c1));
}
}
}
}
q1->clear();
swap(nxt,q1);
} return -;
} void init()
{
int cnt = ;
int id[maxw][maxw];
const int dx[] = {-, , , , };
const int dy[] = { , ,-, , };
int x[maxn];
int y[maxn]; for(int i = ; i < h; i++)
for(int j = ; j < w; j++) {
char ch = maze[i][j];
if(ch != '#'){
x[cnt] = i; y[cnt] = j; id[i][j] = cnt;
if('A'<= ch && ch <= 'C') {t[ch-'A'] = cnt;}
else if('a' <= ch && ch <= 'c') {s[ch-'a'] = cnt; }
cnt++;
}
} for(int i = ; i < cnt; i++){
deg[i] = ;
for(int k = ; k < ; k++) {
int nx = x[i]+dx[k], ny = y[i]+dy[k];
if(maze[nx][ny] != '#')
G[i][deg[i]++] = id[nx][ny];
}
}
if(n<=) {deg[cnt] = ; G[cnt][] = cnt; s[] = t[] = cnt++; }
if(n<=) {deg[cnt] = ; G[cnt][] = cnt; s[] = t[] = cnt++; }
} int main()
{
//freopen("in.txt","r",stdin);
while(~scanf("%d",&w)&&w) {
scanf("%d%d\n",&h,&n);
for(int i = ; i < h; i++)
gets(maze[i]);//G[i-1]
init();
int ans = dBfs();
printf("%d\n",ans);
}
return ;
}
uva 1601 poj 3523 Morning after holloween 万圣节后的早晨 (经典搜索,双向bfs+预处理优化+状态压缩位运算)的更多相关文章
- UVa 1601 || POJ 3523 The Morning after Halloween (BFS || 双向BFS && 降维 && 状压)
		题意 :w*h(w,h≤16)网格上有n(n≤3)个小写字母(代表鬼).要求把它们分别移动到对应的大写字母里.每步可以有多个鬼同时移动(均为往上下左右4个方向之一移动),但每步结束之后任何两个鬼不能占 ... 
- 万圣节后的早晨&&九数码游戏——双向广搜
		https://www.luogu.org/problemnew/show/P1778 https://www.luogu.org/problemnew/show/P2578 双向广搜. 有固定起点终 ... 
- UVa 1601 万圣节后的早晨
		https://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem& ... 
- POJ 3311 Hie with the Pie (BFS+最短路+状态压缩)
		题意:类似于TSP问题,只是每个点可以走多次,求回到起点的最短距离(起点为点0). 分析:状态压缩,先预处理各点之间的最短路,然后sum[i][buff]表示在i点,状态为buff时所耗时...... ... 
- UVA - 658 It's not a Bug, it's a Feature! (隐式图的最短路,位运算)
		隐式的图搜索,存不下边,所以只有枚举转移就行了,因为bug的存在状态可以用二进制表示,转移的时候判断合法可以用位运算优化, 二进制pre[i][0]表示可以出现的bug,那么u&pre[i][ ... 
- UVA 658 状态压缩+隐式图+优先队列dijstla
		不可多得的好题目啊,我看了别人题解才做出来的,这种题目一看就会做的实在是大神啊,而且我看别人博客都看了好久才明白...还是对状态压缩不是很熟练,理解几个位运算用了好久时间.有些题目自己看着别人的题解做 ... 
- <<操作,&0xff以及|的巧妙运用(以POJ3523---The Morning after Halloween(UVa 1601)为例)
		<<表示左移,如a<<1表示将a的二进制左移一位,加一个0,&0xff表示取最后8个字节,如a&0xff表示取a表示的二进制中最后8个数字组成一个新的二进制数, ... 
- 括号序列问题 uva 1626  poj 1141【区间dp】
		首先考虑下面的问题:Code[VS] 3657 我们用以下规则定义一个合法的括号序列: (1)空序列是合法的 (2)假如S是一个合法的序列,则 (S) 和[S]都是合法的 (3)假如A 和 B 都是合 ... 
- POJ 1873 UVA 811 The Fortified Forest (凸包 + 状态压缩枚举)
		题目链接:UVA 811 Description Once upon a time, in a faraway land, there lived a king. This king owned a ... 
随机推荐
- 【转】ssm整合
			http://m.blog.csdn.net/article/details?id=44455235 SSM框架——详细整合教程(Spring+SpringMVC+MyBatis) 发表于2015/3 ... 
- Working Experience - WPF 中 DataGrid 控件的应用
			问题: 添加控件后, 编辑单元格会出现异常 绑定 ItemsSource 属性后, 更新绑定对象的数据, UI 不刷新 如何显示控件中 ComboBox 类型 解决方法: 绑定 ItemsSource ... 
- Git  分支管理  解决冲突
			人生不如意之事十之八九,合并分支往往也不是一帆风顺的. 准备新的feature1分支,继续我们的新分支开发: $ git checkout -b feature1 -- 在feature1分支上修改r ... 
- jmeter后置处理器之正则表达式
			一.基本用法——提取某个值 场景:提取某个值,保存成变量,供后面的接口使用 步骤: 1.运行脚本,从响应结果中查找要提取的值,找到左右边界. 例如要获取“patientInfoId”作为下一个请求的参 ... 
- [TCP/IP]IP协议
			IP数据报 IP是TCP/IP协议族中最核心的协议,所有的TCP.UDP.ICMP.IGMP数据都以IP数据报的格式传输.IP仅提供尽力而为的传输服务,如果发生某种错误,IP会丢失该数据,然后发送IC ... 
- 题解 UVA11354 【Bond】
			并查集+按秩合并 传送门 大意:给出一张n个点m条边的无向图, 每条边有一个权值,有q个询问, 每次给出两个点s.t,找一条路, 使得路径上的边的最大权值最小. 我们可以发现,跑最小生成树会跑挂, 那 ... 
- Codeforces 163E(ac自动机、树状数组)
			要点 显然ac自动机的板子就可以暴力一下答案了 为了优化时间复杂度,考虑套路fail树的dfs序.发现本题需要当前这个尾点加上所有祖先点的个数,考虑使用树状数组差分一下,在父点+1,在子树后-1,每次 ... 
- AtCoder Regular Contest 082 ABCD
			A #include<bits/stdc++.h> using namespace std; ]; int n,m; int main(){ cin>>n>>m; ... 
- block size大小
			1.用tune2fs查看block size大小: 1 2 tune2fs -l /dev/sda1 |grep "Block size" Block size: 1024 2.用 ... 
- CentOS 6.4 中yum命令安装php5.2.17
			最近给公司部署服务器的时候发现他们提供的服务器是centos6.4系统的,装好系统和相关服务httpd,mysql,php,一跑代码,发现php5.3中的zend加密不能用,安装Zend Guard ... 
