HDU 3900 Unblock Me
题目:Unblock Me
链接:Here
题意:一个游戏,看图就基本知道题意了,特殊的是:1. 方块长度为3或2,宽度固定是1。2. 地图大小固定是6*6,从0开始。 3. 出口固定在(6,2)。4. 每一步只能移动一个方块,但可以移动多个单位。5. 必定有解。求红色方块从出口离开的最小步骤数。

思路:
刚开始老老实实的用IDA*做,毫不犹豫的超时,样例都跑不动,剪枝也剪不了,大概知道要状态压缩,具体点就实在想不出来了,百度的题解,感觉刷新了我对状压的认识,用3位二进制表示一个方块的可变信息,然后组成一个LL型数。我写的IDA*,当时想加一个判断当前局面是否遇到过的剪枝,发现要想存状态并且快速找到很困难,现在如果用一个LL型数就可以表示局面,那就简单多了,set就可以解决。
可变信息是指在一个局面开始后会发生变化的信息,相对的不可变信息就比如某个竖立方块的列(在游戏过程中不会发生改变)、长度,因此用二维数组表示状态有很大的冗余,其实在游戏过程中竖立方块会发生变化的就只有行,而地图大小6,所以用3位二进制可以完整地保存,题目最多36/2=18个方块,所以一个LL型可以存下所有方块的信息。
接下来通过简单的位运算可以分离信息,剩下的就是bfs了,难点就在状态压缩,想到以后位置处理就简单了。
AC代码:
#include<stdio.h>
#include<string.h>
#include<stdlib.h>
#include<math.h>
#include<set>
#include<map>
#include<list>
#include<stack>
#include<queue>
#include<vector>
#include<string>
#include<iostream>
#include<algorithm>
using namespace std;
#define lson rt<<1
#define rson rt<<1|1
#define N 20020
#define M 100010
#define Mod 1000000007
#define LL long long
#define INF 0x7fffffff
#define FOR(i,f_start,f_end) for(int i=f_start;i<=f_end;i++)
#define For(i,f_start,f_end) for(int i=f_start;i<f_end;i++)
#define REP(i,f_end,f_start) for(int i=f_end;i>=f_start;i--)
#define Rep(i,f_end,f_start) for(int i=f_end;i>f_start;i--)
#define MT(x,i) memset(x,i,sizeof(x))
#define gcd(x,y) __gcd(x,y)
const double PI = acos(-); struct Node
{
bool type; //竖的 1 ,横的 0
int pos; //竖的记录行,横的记录列
int len; //长度
}v[]; int cal(LL s,int pos)
{
return (int)( ( s & (7LL << (pos * )) ) >> (pos * ) );
} int n,red; bool ok(LL s)
{
int rl = cal(s,red), rr = rl + v[red].len;
for(int i=;i<n;i++)
{
if(v[i].type) continue;
if(v[i].pos <= rr) continue;
int il = cal(s,i), ir = il + v[i].len;
if(ir < || il > ) continue;
return false;
}
return true;
} bool check(LL s,int pre)
{
int pl = cal(s,pre), pr = pl + v[pre].len;
for(int i=;i<n;i++)
{
if(i == pre) continue;
if(v[i].type == v[pre].type)
{
if(v[i].pos!=v[pre].pos) continue;
int il = cal(s,i), ir = il + v[i].len;
if(ir<pl || pr<il) continue;
}
else
{
int il = cal(s,i), ir = il + v[i].len;
if( pr < v[i].pos || pl > v[i].pos || il > v[pre].pos || ir < v[pre].pos) continue;
}
return false;
}
return true;
} void mov(LL &s,int pos,int x)
{
s &= ( ~( 7LL << (*pos) ) );
s |= ( (LL)x << (*pos) );
} set<LL> mp;
queue<pair<LL,int> > q;
int bfs(LL s)
{
if(ok(s)) return ;
while(q.size()) q.pop();
mp.clear();
q.push(make_pair(s,) );
while(q.size())
{
s = q.front().first;
int d = q.front().second + ;
q.pop();
for(int i=;i<n;i++)
{
int p = cal(s,i);
for(int j=;p-j>=;j++)
{
LL u = s;
mov(u,i,p-j);
if(check(u,i)==false) break;
if(mp.find(u)!=mp.end()) continue;
if(ok(u)) return d+;
mp.insert(u);
q.push(make_pair(u,d));
}
for(int j=;p+v[i].len+j<=;j++)
{
LL u = s;
mov(u,i,p+j);
if(check(u,i) == false) break;
if(mp.find(u)!=mp.end()) continue;
if(ok(u)) return d+;
mp.insert(u);
q.push(make_pair(u,d));
}
}
}
return -;
} int main()
{
while(~scanf("%d",&n))
{
LL s = ;
int x,y,x1,y1;
For(i,,n){
scanf("%*d%d%d%d%d",&x,&y,&x1,&y1);
if(x==x1){ // su
v[i].type = ;
v[i].pos = x;
v[i].len = y1-y;
s |= ( (LL)y << (i*) );
}
else
{
v[i].type = ;
v[i].pos = y;
v[i].len = x1-x;
s |= ( (LL)x << (i*) );
}
}
scanf("%d",&red);
printf("%d\n",bfs(s));
}
return ;
}
HDU 3900 Unblock Me的更多相关文章
- HDU 5643 King's Game 打表
King's Game 题目连接: http://acm.hdu.edu.cn/showproblem.php?pid=5643 Description In order to remember hi ...
- HDOJ 2111. Saving HDU 贪心 结构体排序
Saving HDU Time Limit: 3000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total ...
- 【HDU 3037】Saving Beans Lucas定理模板
http://acm.hdu.edu.cn/showproblem.php?pid=3037 Lucas定理模板. 现在才写,noip滚粗前兆QAQ #include<cstdio> #i ...
- hdu 4859 海岸线 Bestcoder Round 1
http://acm.hdu.edu.cn/showproblem.php?pid=4859 题目大意: 在一个矩形周围都是海,这个矩形中有陆地,深海和浅海.浅海是可以填成陆地的. 求最多有多少条方格 ...
- HDU 4569 Special equations(取模)
Special equations Time Limit:1000MS Memory Limit:32768KB 64bit IO Format:%I64d & %I64u S ...
- HDU 4006The kth great number(K大数 +小顶堆)
The kth great number Time Limit:1000MS Memory Limit:65768KB 64bit IO Format:%I64d & %I64 ...
- HDU 1796How many integers can you find(容斥原理)
How many integers can you find Time Limit:5000MS Memory Limit:32768KB 64bit IO Format:%I64d ...
- hdu 4481 Time travel(高斯求期望)(转)
(转)http://blog.csdn.net/u013081425/article/details/39240021 http://acm.hdu.edu.cn/showproblem.php?pi ...
- HDU 3791二叉搜索树解题(解题报告)
1.题目地址: http://acm.hdu.edu.cn/showproblem.php?pid=3791 2.参考解题 http://blog.csdn.net/u013447865/articl ...
随机推荐
- 5255 -- 【FJOI2016】神秘数
5255 -- [FJOI2016]神秘数 Description 一个可重复数字集合\(S\) 的神秘数定义为最小的不能被 \(S\) 的子集的和表示的正整数.例如: \(S = {1,1,1,4, ...
- 基于diango简易BLOG图片预览
注册界面 登陆界面 博客主界面 个人站点主页 文章详情页 个人博客后台 添加文章页 修改密码页
- python 中的__init__.py的用法与个人理解
使用Python模块常见的情况是,事先写好A.py文件,需要import B.py文件时,先拷贝到当前目录,然后再import 这样的做法在程序量较小的情况下是可行的,如果程序交互复杂程度稍高,就很费 ...
- ORACLE直方图(10g)
为什么需要直方图 ?当表中一列数据比较的值分布比较均匀时,optimzer可以很好的通过最大值,最小值和NDV(唯一值的个数),就可以判断出cardinality.对于cardinality越精确,o ...
- node访问oracledb的环境搭建
关于安装oracleDB环境官网说明地址: https://oracle.github.io/node-oracledb/INSTALL.html 环境搭建所需软件和文档的压缩包 链接: https: ...
- 从零开始搭建django前后端分离项目 系列四(实战之实时进度)
本项目实现了任务执行的实时进度查询 实现方式 前端websocket + 后端websocket + 后端redis订阅/发布 实现原理 任务执行后,假设用变量num标记任务执行的进度,然后将num发 ...
- X86-64寄存器和栈帧--牛掰降解汇编函数寄存器相关操作
X86-64寄存器和栈帧 概要 说到x86-64,总不免要说说AMD的牛逼,x86-64是x86系列中集大成者,继承了向后兼容的优良传统,最早由AMD公司提出,代号AMD64:正是由于能向后兼容,AM ...
- JSOUP如何优秀的下载JPEG等二进制图像
引言 JSOUP默认是不支持解析JPEG等二进制图像的,解决方法也很简单,只需要加上Jsoup.ignoreContentType(true)这一行代码就可以.关于这一点的原因,来看看官方API说明. ...
- win2016 配置IIS 和mysql5.7 迁移数据表的两个小坑
今天配置一整天,就IIS都装了一整天,都是没办法安装.net3.5的问题. 最后解决办法:https://help.aliyun.com/knowledge_detail/38203.html?spm ...
- Java面试总结之AIO与NIO
1.Java NIO 是一种同步非阻塞的I/O模型 将多个IO的阻塞复用到同一个select的阻塞上,从而使得系统在单线程的情况下处理多个客户端请求. NIO三个核心对象:通道(Channel).缓冲 ...