POJ - 1324 Holedox Moving (状态压缩+BFS/A*)
有一个n*m(1<=n,m<=20)的网格图,图中有k堵墙和有一条长度为L(L<=8)的蛇,蛇在移动的过程中不能碰到自己的身体。求蛇移动到点(1,1)所需的最小步数。
显然用8个(x,y)来表示蛇的状态是不现实的(用哈希也很难存下,要么爆内存,要么超时),所以首先应当进行状态压缩。可以发现蛇的身体是连续的,因此可以用一个表示方向的向量来储存蛇身体的每个部分在它上个部分的哪个方向(只有头部用(x,y)表示),这样总状态数就变成了n*m*(2^((L-1)*2)),在可接受范围内了。
可优化的部分:
1.可以在图的边界上放上一层墙,避免出界判定。
2.如果在碰撞判定时把“蛇身”和“墙”等同的话,为了防止在撤销蛇身所造成的影响的同时把墙也一并消除,可以用b[x][y]++,--的方法来代替b[x][y]=1。
AC代码:(编码和解码真是个体力活)
#include<cstdio>
#include<cstring>
#include<queue>
using namespace std;
typedef long long ll;
const int N=+;
const int dx[]= {,,-,};
const int dy[]= {-,,,};
struct P {int x,y;} p[N],r[N];
int n,m,l,k,b[N][N],d[N][N][(<<)+],S,ka;
struct D {int x,y,S;}; int bfs() {
queue<D> q;
memset(d,-,sizeof d);
d[p[].x][p[].y][S]=,q.push({p[].x,p[].y,S});
while(!q.empty()) {
int x=q.front().x,y=q.front().y,S=q.front().S;
q.pop();
if(x==&&y==)return d[x][y][S];
r[]= {x,y};
for(int i=; i<l; ++i) {
int t=S>>((l-i-)*)&;
r[i]= {r[i-].x+dx[t],r[i-].y+dy[t]};
}
for(int i=; i<l; ++i)b[r[i].x][r[i].y]++;
for(int i=; i<; ++i) {
int xx=x+dx[i],yy=y+dy[i],SS=S>>|((i^)<<((l-)*));
if(!b[xx][yy]&&!~d[xx][yy][SS])d[xx][yy][SS]=d[x][y][S]+,q.push({xx,yy,SS});
}
for(int i=; i<l; ++i)b[r[i].x][r[i].y]--;
}
return -;
} int main() {
while(scanf("%d%d%d",&n,&m,&l)&&n) {
printf("Case %d: ",++ka);
for(int i=; i<l; ++i)scanf("%d%d",&p[i].x,&p[i].y);
S=;
for(int i=; i<l; ++i)
for(int j=; j<; ++j)if(p[i-].x+dx[j]==p[i].x&&p[i-].y+dy[j]==p[i].y) {
S=S<<|j;
break;
}
memset(b,,sizeof b);
for(int i=; i<=n+; ++i)b[i][]=b[i][m+]=;
for(int i=; i<=m+; ++i)b[][i]=b[n+][i]=;
scanf("%d",&k);
while(k--) {
int x,y;
scanf("%d%d",&x,&y);
b[x][y]=;
}
printf("%d\n",bfs());
}
return ;
}
bfs
也可以进一步用A*优化,以每个点到点(1,1)的距离为估价函数进行转移,速度大幅提升。
一般A*中优先队列的结点要额外设一个属性g来表示已走过的距离,但也可以省去这个g,用d[x][y][S]来代替g,这样如果要把bfs改成A*的话,只要写出求每个点到点(1,1)距离的bfs代码,然后把原来的bfs代码中的queue换成priority_queue,front换成top就ok了,非常方便。
另外可以优化的一个地方是如果一个点不能到达(1,1),那么直接返回-1就行,不用再浪费时间跑A*了。
#include<cstdio>
#include<cstring>
#include<queue>
using namespace std;
typedef long long ll;
const int N=+;
const int dx[]= {,,-,};
const int dy[]= {-,,,};
struct P {int x,y;} p[N],r[N];
int n,m,l,k,b[N][N],d[N][N][(<<)+],S,h[N][N],ka;
struct D {
int x,y,S;
bool operator<(const D& b)const {return d[x][y][S]+h[x][y]>d[b.x][b.y][b.S]+h[b.x][b.y];}
}; void bfs() {
queue<P> q;
memset(h,-,sizeof h);
h[][]=,q.push({,});
while(!q.empty()) {
int x=q.front().x,y=q.front().y;
q.pop();
for(int i=; i<; ++i) {
int xx=x+dx[i],yy=y+dy[i];
if(!~h[xx][yy]&&!b[xx][yy])h[xx][yy]=h[x][y]+,q.push({xx,yy});
}
}
} int Astar() {
if(!~h[p[].x][p[].y])return -;
priority_queue<D> q;
memset(d,-,sizeof d);
d[p[].x][p[].y][S]=,q.push({p[].x,p[].y,S});
while(!q.empty()) {
int x=q.top().x,y=q.top().y,S=q.top().S;
q.pop();
if(x==&&y==)return d[x][y][S];
r[]= {x,y};
for(int i=; i<l; ++i) {
int t=S>>((l-i-)*)&;
r[i]= {r[i-].x+dx[t],r[i-].y+dy[t]};
}
for(int i=; i<l; ++i)b[r[i].x][r[i].y]++;
for(int i=; i<; ++i) {
int xx=x+dx[i],yy=y+dy[i],SS=S>>|((i^)<<((l-)*));
if(!b[xx][yy]&&!~d[xx][yy][SS])d[xx][yy][SS]=d[x][y][S]+,q.push({xx,yy,SS});
}
for(int i=; i<l; ++i)b[r[i].x][r[i].y]--;
}
return -;
} int main() {
while(scanf("%d%d%d",&n,&m,&l)&&n) {
printf("Case %d: ",++ka);
for(int i=; i<l; ++i)scanf("%d%d",&p[i].x,&p[i].y);
S=;
for(int i=; i<l; ++i)
for(int j=; j<; ++j)if(p[i-].x+dx[j]==p[i].x&&p[i-].y+dy[j]==p[i].y) {
S=S<<|j;
break;
}
memset(b,,sizeof b);
for(int i=; i<=n+; ++i)b[i][]=b[i][m+]=;
for(int i=; i<=m+; ++i)b[][i]=b[n+][i]=;
scanf("%d",&k);
while(k--) {
int x,y;
scanf("%d%d",&x,&y);
b[x][y]=;
}
bfs();
printf("%d\n",Astar());
}
return ;
}
A*
POJ - 1324 Holedox Moving (状态压缩+BFS/A*)的更多相关文章
- POJ 1324 Holedox Moving (状压BFS)
POJ 1324 Holedox Moving (状压BFS) Time Limit: 5000MS Memory Limit: 65536K Total Submissions: 18091 Acc ...
- poj 1324 Holedox Moving
poj 1324 Holedox Moving 题目地址: http://poj.org/problem?id=1324 题意: 给出一个矩阵中,一条贪吃蛇,占据L长度的格子, 另外有些格子是石头, ...
- POJ 1324 Holedox Moving 搜索
题目地址: http://poj.org/problem?id=1324 优先队列---A*的估价函数不能为蛇头到(1,1)的距离,这样会出错. 看了discuss,有大神说这题A*的估价函数为BFS ...
- POJ 3411 Paid Roads (状态压缩+BFS)
题意:有n座城市和m(1<=n,m<=10)条路.现在要从城市1到城市n.有些路是要收费的,从a城市到b城市,如果之前到过c城市,那么只要付P的钱, 如果没有去过就付R的钱.求的是最少要花 ...
- POJ 1753 Flip Game(状态压缩+BFS)
题目网址:http://poj.org/problem?id=1753 题目: Flip Game Description Flip game is played on a rectangular 4 ...
- poj 1324 状态压缩+bfs
http://poj.org/problem?id=1324 Holedox Moving Time Limit: 5000MS Memory Limit: 65536K Total Submis ...
- 胜利大逃亡(续)(状态压缩bfs)
胜利大逃亡(续) Time Limit: 4000/2000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) Total S ...
- POJ 1753 Flip Game (状态压缩 bfs+位运算)
Flip game is played on a rectangular 4x4 field with two-sided pieces placed on each of its 16 square ...
- hdu 3681 Prison Break(状态压缩+bfs)
Problem Description Rompire . Now it’s time to escape, but Micheal# needs an optimal plan and he con ...
随机推荐
- MySQL数据库(9)_MySQL数据库常用操作命令
注:刚安装好的MySql包含一个含空密码的root帐户和一个匿名帐户,这是很大的安全隐患,对于一些重要的应用我们应将安全性尽可能提高,在这里应把匿名帐户删除. root帐户设置密码,可用如下命令进行: ...
- Oracle索引表
索引组织表(Index-Organized Table)是按B-树的结构来组织和存储数据的.与标准表中的数据时无序存放的不同,索引表中数据按主键值有序存储. 叶子节点中存放的是表的主键值与所有非主键值 ...
- yii异常处理架构
使用方法: use \yii\base\ErrorException; try { exec("curl http://xxx",$out,$retno); if(0 !== $r ...
- Android摄像头测量尺(Advanced Ruler Pro)使用方法
http://www.cnblogs.com/sinojelly/archive/2010/08/13/1799341.html Advanced Ruler Pro是一个Android手机应用程序, ...
- 物理分辨率与逻辑分辨率,pt与px
有些小伙伴们,在使用chrome的移动端调试工具调试网页的时候,会发现iphone6上的尺寸为375*667,不禁差异,iphone6的分辨率不是750*1334吗? 实际上调试器上的大小单位不是px ...
- 创建Android守护进程(底层服务)【转】
本文转载自:https://blog.csdn.net/myfriend0/article/details/80016739 创建Android守护进程(底层服务) 前言 Android底层服务,即运 ...
- 如何在windows10环境下安装Pytorch-0.4.1版本
开始是按照教程:https://blog.csdn.net/xiangxianghehe/article/details/80103095 安装了Pytorch0.4.0,但是安装后发现在import ...
- WINDOWS下好用的MongoDB 3.0以上客户端工具: NoSql Manager
WINDOWS下好用的MongoDB 3.0以上客户端工具: NoSql Manager https://www.mongodbmanager.com/download
- 搭建配置cacti,采集信息监控
安装cactilamp环境[iyunv@Cacti ~]#service iptables stop //关闭防火墙服务[iyunv@Cacti ~]#chkconfig iptables off / ...
- linux基础(3)-java安装
安装jdk1.8 [root@spark1 usr]# mkdir java #创建java目录 通过WinSCP将jdk-8u77-linux-x64.tar.gz传到/usr/java目录下 [r ...