题目链接

有一个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*)的更多相关文章

  1. POJ 1324 Holedox Moving (状压BFS)

    POJ 1324 Holedox Moving (状压BFS) Time Limit: 5000MS Memory Limit: 65536K Total Submissions: 18091 Acc ...

  2. poj 1324 Holedox Moving

    poj 1324 Holedox Moving 题目地址: http://poj.org/problem?id=1324 题意: 给出一个矩阵中,一条贪吃蛇,占据L长度的格子, 另外有些格子是石头, ...

  3. POJ 1324 Holedox Moving 搜索

    题目地址: http://poj.org/problem?id=1324 优先队列---A*的估价函数不能为蛇头到(1,1)的距离,这样会出错. 看了discuss,有大神说这题A*的估价函数为BFS ...

  4. POJ 3411 Paid Roads (状态压缩+BFS)

    题意:有n座城市和m(1<=n,m<=10)条路.现在要从城市1到城市n.有些路是要收费的,从a城市到b城市,如果之前到过c城市,那么只要付P的钱, 如果没有去过就付R的钱.求的是最少要花 ...

  5. POJ 1753 Flip Game(状态压缩+BFS)

    题目网址:http://poj.org/problem?id=1753 题目: Flip Game Description Flip game is played on a rectangular 4 ...

  6. poj 1324 状态压缩+bfs

    http://poj.org/problem?id=1324 Holedox Moving Time Limit: 5000MS   Memory Limit: 65536K Total Submis ...

  7. 胜利大逃亡(续)(状态压缩bfs)

    胜利大逃亡(续) Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total S ...

  8. 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 ...

  9. hdu 3681 Prison Break(状态压缩+bfs)

    Problem Description Rompire . Now it’s time to escape, but Micheal# needs an optimal plan and he con ...

随机推荐

  1. python之数据的序列化

    参考博客:http://www.cnblogs.com/yyds/p/6563608.html 数据的序列化功能表 json.dumps() 将python数据类型转换为(json)字符串 json. ...

  2. springMVC中使用 RequestBody 及 Ajax POST请求 415 (Unsupported Media Type)

    使用POST请求的时候一直报错: Ajax 未设置 contentType 时会报 415 . 后台 RequestBody  承接前台参数,故对参数data的要求为“必传”“JSON”,否则会报40 ...

  3. MySQL数据库(2)_MySQL数据库和数据库表操作语句

    一.关于数据库操作的sql语句 -- .创建数据库(在磁盘上创建一个对应的文件夹) create database [if not exists] db_name [character set xxx ...

  4. Apache Shiro:【1】Shiro基础及Web集成

    Apache Shiro:[1]Shiro基础及Web集成 Apache Shiro是什么 Apache Shiro是一个强大且易于使用的Java安全框架,提供了认证.授权.加密.会话管理,与spri ...

  5. Linux Shell基础 Bash常见命令 echo命令

    概述 shell中常见的命令echo. 输出命令:echo echo命令的输出内容如果没有特殊含义,则将原内容输出到屏幕:如果输出内容有特殊含义,则输出打印其含义. 命令格式如下: [root@loc ...

  6. vi高级命令集锦

    .交换两个字符位置 xp .上下两行调换 ddp .把文件内容反转 :g/^/m0/ (未通过) .上下两行合并 J .删除所有行 dG .从当前位置删除到行尾 d$ .从当前位置复制到行尾 y$ 如 ...

  7. 【Head First Servlets and JSP】笔记19:JavaBeans与JSP动作元素(<jsp:setProperty.....>、<jsp:getProperty.....>)

    内容来自imooc. 1.什么是JSP动作元素 2.在JSP页面中如何使用Javabeans <jsp:......>表示这是一个JSP动作元素 3.使用JSP动作元素创建JavaBean ...

  8. 非root权限的linux用法添加工作路径

    修改~目录的bashrc文件: 1.cd 到~目录. 2.ls -a ,bashrc文件是隐藏的. 3.vim .bashrc;export PATH=$PAHT:要添加的工作路径. 4.source ...

  9. [RK3288][Android6.0] 调试笔记 --- user版本默认显示开发者选项【转】

    本文转载自:https://blog.csdn.net/kris_fei/article/details/70157137 Platform: ROCKCHIPOS: Android 6.0Kerne ...

  10. RHEL 7 安装 ngnix

    安装ngnix yum install -y make apr* autoconf automake curl curl-devel gcc gcc-c++ gtk+-devel zlib-devel ...