P4289 [HAOI2008]移动玩具(bfs)
双向bfs+状态压缩+记忆化搜索
双向bfs用于对bfs的优化,每次找到可扩展节点少的一边进行一次bfs,找到的第一个互相接触的点即为最短路径
矩阵范围仅4*4大小,我们容易想到用二进制数压缩其状态,利于求解。
既然转成二进制,大小又<2^17,那么可以再加数组进行记忆化
不要忘了起点终点相同时的特判qwq
- #include<iostream>
- #include<cstdio>
- #include<queue>
- #include<cstring>
- using namespace std;
- struct data{int zip,step;};
- int ans,rec1[],rec2[]; //rec:记忆化用
- queue <data> h[]; //分别代表从起点/终点开始的bfs队列
- inline void check(int to,int p,data x){ //检查该点是否符合条件
- if(rec1[to]!=-){
- if(rec2[to]!=p) ans=rec1[to]+x.step+; //如果该点已被对面搜到,那么已经得出最优解
- }else{
- rec1[to]=x.step+,rec2[to]=p;
- h[p].push((data){to,x.step+});
- }
- }
- void output(int t){ //检查用,将10进制数转回4*4的二进制矩阵
- cout<<"---\n";
- int h[];
- for(int k=t,i=;i>=;--i) h[i]=k&,k>>=;
- for(int i=;i<;++i){
- for(int j=i*;j<=i*+;++j) cout<<h[j];
- cout<<endl;
- }
- cout<<"---\n";
- }
- inline void bfs(){
- int p=(h[].size()>h[].size()),now=(h[p].front()).step; //找到可扩展节点少的一边,并且只扩展一层
- while(!h[p].empty()){
- data x=h[p].front();
- if(x.step!=now||ans) break;
- h[p].pop();
- int k=,to;
- for(int i=;i<;i<<=,++k){ //用二进制数表示转移过程
- if(!(x.zip&i)) continue;
- if(k%!=&&(!(x.zip&(i<<)))){ //向左
- to=x.zip-i+(i<<);
- check(to,p,x);
- }
- if(k%!=&&(!(x.zip&(i>>)))){ //向右
- to=x.zip-i+(i>>);
- check(to,p,x);
- }
- if(k>&&(!(x.zip&(i>>)))){ //向下
- to=x.zip-i+(i>>);
- check(to,p,x);
- }
- if(k<&&(!(x.zip&(i<<)))){ //向上
- to=x.zip-i+(i<<);
- check(to,p,x);
- }
- }
- }
- }
- int main(){
- memset(rec1,-,sizeof(rec1));
- char q[]; int tot1=,tot2=;
- for(int i=;i<=;++i){ //矩阵转成十进制数
- cin>>q;
- for(int j=;j<=;++j) tot1+=q[j]-,tot1<<=;
- }tot1>>=; //注意最后要右移一位
- h[].push((data){tot1,}); rec1[tot1]=; rec2[tot1]=;
- for(int i=;i<=;++i){
- cin>>q;
- for(int j=;j<=;++j) tot2+=q[j]-,tot2<<=;
- }tot2>>=;
- h[].push((data){tot2,}); rec1[tot2]=; rec2[tot2]=;
- while(!ans&&tot1!=tot2) bfs(); //注意特判
- printf("%d",ans);
- return ;
- }
P4289 [HAOI2008]移动玩具(bfs)的更多相关文章
- 【BZOJ1054】[HAOI2008]移动玩具 BFS
[BZOJ1054][HAOI2008]移动玩具 Description 在一个4*4的方框内摆放了若干个相同的玩具,某人想将这些玩具重新摆放成为他心中理想的状态,规定移动 时只能将玩具向上下左右四个 ...
- [BZOJ1054][HAOI2008]移动玩具 bfs+hash
1054: [HAOI2008]移动玩具 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 2432 Solved: 1355[Submit][Stat ...
- bzoj 1054: [HAOI2008]移动玩具 bfs
1054: [HAOI2008]移动玩具 Time Limit: 10 Sec Memory Limit: 162 MB[Submit][Status][Discuss] Description 在 ...
- P4289 [HAOI2008]移动玩具
传送门 广搜 4*4 的方阵只有 0 和 1 显然可以状态压缩 (如样例的开始状态压缩后就是1111000011100010) 为了加快速度用了双向广搜(顺便学了一下双向广搜) 双向广搜顾名思义 就是 ...
- luogu P4289 [HAOI2008]移动玩具
传送门 这道题可以二进制记录状态搜索 也可以做以下考虑 若一个棋子要移动到另一个位置上去,则步数为两点的曼哈顿距离(横坐标差的绝对值+纵坐标差的绝对值),因为假设路径上有其他的棋子,可以通过移动其他棋 ...
- P4289 【一本通提高篇广搜的优化技巧】[HAOI2008]移动玩具
[HAOI2008]移动玩具 题目描述 在一个 4 × 4 4\times4 4×4 的方框内摆放了若干个相同的玩具,某人想将这些玩具重新摆放成为他心中理想的状态,规定移动时只能将玩具向上下左右四个方 ...
- bzoj1054: [HAOI2008]移动玩具
hash+bfs:要注意特殊情况.(似乎连sort.lower_bound都不用数据小直接判重了... #include<cstdio> #include<cstring> # ...
- BZOJ 1054 [HAOI2008]移动玩具
1054: [HAOI2008]移动玩具 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 1388 Solved: 764[Submit][Statu ...
- 1054: [HAOI2008]移动玩具
1054: [HAOI2008]移动玩具 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 1272 Solved: 690[Submit][Statu ...
随机推荐
- Oracle管理监控之Oracle用户权限
数据字典 1.动态数据字典是以v$xxx开始的数据字典,在数据库中约有150个左右,这些数据字典反映数据库动态运行状况,在不同时间查询会得到不同的结果. 2.DBA数据字典是以DBA_xxx表示,该数 ...
- free详解
用法: [oracle@server36 ~]$ free -help free: invalid option -- h usage: free [-b|-k|-m|-g] [-l] [-o] [- ...
- Docker处理日志的方法&日志收集工具比较
测试logstash:docker run -it mylogstash:0.1.0 logstash -e 'input{stdin{}}output{stdout{codec=>rubyde ...
- 洛谷P1084 疫情控制 [noip2012] 贪心+树论+二分答案 (还有个小bugQAQ
正解:贪心+倍增+二分答案 解题报告: 正好想做noip的题目然后又想落实学长之前讲的题?于是就找上了这题 其实之前做过,70,然后实在细节太多太复杂就不了了之,现在再看一遍感觉又一脸懵了... 从标 ...
- post方式提交数据
<!DOCTYPE HTML><html> <head> <meta charset="utf-8" /> ...
- mysql 权限管理 针对表的字段 级别 授权 columns_priv表
针对Mike账号 db1库下面的t1表的 id,name字段授予select权限,age字段授予update权限 授权格式 select(要授权的字段,要授权的字段) 用户括号 括起来 .updat ...
- python 面向对象 issubclass
判断是否 他的父类 class Foo(object): pass obj = Foo() class Boo(Foo): pass class Coo(Boo): pass obj = Boo() ...
- CentOS工作内容(三)配置网络IP地址
CentOS工作内容(三)配置网络IP地址 用到的快捷键 tab 自动补齐(有不知道的吗) ctrl+a 移动到当前行的开头(a ahead) ctrl+u 删除(剪切)此处至开始所有内容 vim 末 ...
- java-JProfiler(五)-监控性能
原文地址:http://blog.csdn.net/chendc201/article/details/22897999 一.基础认识 1. 在Live Memory视图里右击相关类,选中Mark C ...
- Java-小技巧-002-String 转 long
1.转化 long l = Long.parseLong([String]); 相当于 long l = Long.parseLong([String],10); long l = Long.valu ...