Eight HDU - 1043 (双向BFS)
记得上人工智能课的时候老师讲过一个A*算法,计算估价函数(f[n]=h[n]+g[n])什么的,感觉不是很好理解,百度上好多都是用逆向BFS写的,我理解的逆向BFS应该是从终点状态出发,然后把每一种状态打表列举出来,最后O(1)查询就可以了。这种办法确实挺好,但是不会....。
这位大佬用的双向BFS https://blog.csdn.net/qq_41670466/article/details/84110090,挺好理解的,但是注释什么的比较少,也没有过多的介绍思路,所以我想借助这篇blog简单的介绍一下这个题目的双向BFS的思路。
双向BFS就是终点状态(从后向前)和起始状态(从前向后)一起寻找,当且仅当而且碰头时,就是答案了。

针对本题来说,初始状态就是输入的转态,终点状态都是一样的123456780(x用0来代替),但是这个状态怎么表示呢?康托展开 (不会的可以点开看一下)。状态表示解决完了,接下来我们看一下转态转移。

当x处在0 3 6这三个位置时,不可以向左移动,当x处于2 5 8这三个位置时,不可以向右移动,当x处于0 1 2这三个位置时,不可以向上移动,处于6 7 8这三个位置时,不可以向下移动。 对于本题样例,s[]=2 3 4 1 5 x 7 6 8。,x是处于5这个位置,如果向上移动,就可以看成s[5]和s[5-3]交换了一下,向下移动可以看成s[5]和s[5+3]交换了一下,
向左s[5]和s[5-1]向右同理....这样就实现了状态之间的转移。
路径的记录。维护两个数组char 和int ,char 用来记录向哪移动了,int 用来记录上一个状态的下标。
最后还要加一个特判,问题是否有解跟逆序对有关,如果逆序对是奇数就有解,否则就无解具体为什么跟线性代数有关吧~~我也不太懂(待解释)。
具体实现都在code中了....
#include<bits/stdc++.h>
using namespace std;
const int maxn=5e5+;
string a;
int ha[]={,,,,,,,,};//从8到0对应的阶乘
int vis[maxn],vis2[maxn];
struct node {
int num;
char ch;
}pre[maxn];//write path
struct stu {
string s;//当前串
int num;//x的位置
}e;//save now state
int direction[]={-,,-,};//udlr
string s1="udlr",s2="durl";
int ct(string s){//当前串对应的康托值
int sum=;
for(int i=;i<;i++){
int k=;
for(int j=i+;j<;j++){
if(s[j]<s[i]) k++;
}
sum+=k*ha[i];
}
return sum;
}
void writhpath(int x){
if(pre[x].num==-) return ;
writhpath(pre[x].num);
printf("%c",pre[x].ch);
}
void bfs(){
queue<stu> que1,que2;
int q=ct(e.s);
vis[q]=;
stu f,g;
f.s="";
f.num=;
int p=ct(f.s);
vis2[p]=;
pre[].num=-;pre[].num=-; int num=;
que1.push(e);que2.push(f);
while(que1.size()&&que2.size()){
f=que1.front();que1.pop();
p=ct(f.s);
if(vis2[p]){
writhpath(vis[p]);
int k=vis2[p];
while(pre[k].num!=-) {
printf("%c",pre[k].ch);
k=pre[k].num;
}
cout<<endl;
return ;
}
else{
for(int i=;i<;i++){
if(i==&&f.num<) continue ;
if(i==&&f.num>) continue ;
if(i==&&f.num%==) continue ;
if(i==&&f.num%==) continue ;
int dx=f.num+direction[i];g=f;
swap(g.s[f.num],g.s[dx]);
q=ct(g.s);
if(vis[q]) continue ;
vis[q]=++num;g.num=dx;
que1.push(g);
pre[num].num=vis[p];pre[num].ch=s1[i];
}
}
f=que2.front();que2.pop();
p=ct(f.s);
if(vis[p]){
writhpath(vis[p]);
int k=vis2[p];
while(pre[k].num!=-) {
printf("%c",pre[k].ch);
k=pre[k].num;
}
cout<<endl;
return ;
}
else{
for(int i=;i<;i++){
if(i==&&f.num<) continue ;
if(i==&&f.num>) continue ;
if(i==&&f.num%==) continue ;
if(i==&&f.num%==) continue ;
int dx=f.num+direction[i];g=f;
swap(g.s[f.num],g.s[dx]);
q=ct(g.s);
if(vis2[q]) continue ;
vis2[q]=++num;g.num=dx;
que2.push(g);
pre[num].num=vis2[p];pre[num].ch=s2[i];
}
}
}
puts("unsolvable");
}
int main(){
while(getline(cin,a)){
string c="";
int n=a.size(),pos=,j=;
for(int i=;i<n;i++){
if(a[i]==' ') continue ;
if(a[i]=='x'){
c+='';pos=j;
}
else {
c+=a[i];j++;
}
}
int k=;e.num=pos;e.s=c;
for (int i=;i<;i++){
if (e.s[i]=='')continue;
for (int j = ;j<i;j++){
if (e.s[j] == '')continue;
if (e.s[j]>e.s[i])k++;
}
}
if(k&) {
puts("unsolvable");
}
else {
memset(vis,,sizeof vis);
memset(vis2,,sizeof vis2);
bfs();
}
}
return ;
}
Eight HDU - 1043 (双向BFS)的更多相关文章
- HDU - 3085 双向BFS + 技巧处理 [kuangbin带你飞]专题二
题意:有两只鬼,一个男孩女孩被困在迷宫中,男孩每秒可以走三步,女孩只能1步,鬼可以两步且可以通过墙.问男孩女孩是否可以在鬼抓住他们之前会合? 注意:每秒开始鬼先移动,然后两人开始移动. 思路:以男孩和 ...
- hdu 3085(双向bfs)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3085 思路:双向广搜,每次从M出发,搜三步,从G出发,搜一步,然后就是判断是否走到对方已经走过的格子, ...
- HDU 1043 Eight BFS
题意:就是恢复成1,2,3,4,5,6,7,8,0; 分析:暴力BFS预处理,所有路径,用康拓展开判重,O(1)打印 93ms 还是很快的 #include <iostream> #inc ...
- HDU 1043 Eight (BFS·八数码·康托展开)
题意 输出八数码问题从给定状态到12345678x的路径 用康托展开将排列相应为整数 即这个排列在全部排列中的字典序 然后就是基础的BFS了 #include <bits/stdc++.h ...
- HDU 1043 Eight(双向BFS+康托展开)
http://acm.hdu.edu.cn/showproblem.php?pid=1043 题意:给出一个八数码,求出到达指定状态的路径. 思路:路径寻找问题.在这道题里用到的知识点挺多的.第一次用 ...
- Eight (HDU - 1043|POJ - 1077)(A* | 双向bfs+康拓展开)
The 15-puzzle has been around for over 100 years; even if you don't know it by that name, you've see ...
- hdu 1043 Eight(双向bfs)
题意:经典八数码问题 思路:双向bfs ps:还有a*算法(还不会)等解法. 代码: #include<iostream> #include<stdio.h> #include ...
- HDU 3085 Nightmare II 双向bfs 难度:2
http://acm.hdu.edu.cn/showproblem.php?pid=3085 出的很好的双向bfs,卡时间,普通的bfs会超时 题意方面: 1. 可停留 2. ghost无视墙壁 3. ...
- 2017多校第10场 HDU 6171 Admiral 双向BFS或者A*搜索
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6171 题意: 给你一个高度为6的塔形数组,你每次只能将0与他上下相邻的某个数交换,问最少交换多少次可以 ...
随机推荐
- Selenium系列(九) - 针对alert窗口的处理(警告框、确认框、对话框)
如果你还想从头学起Selenium,可以看看这个系列的文章哦! https://www.cnblogs.com/poloyy/category/1680176.html 其次,如果你不懂前端基础知识, ...
- shell脚本中的if条件语句介绍和使用案例
#前言:在生产工作中if条件语句是最常使用的,如使用来判断服务状态,监控服务器的CPU,内存,磁盘等操作,所以我们需要熟悉和掌握if条件语句. #简介 if条件语句,简单来说就是:如果,那么.有if单 ...
- 答应我,不会这些概念,简历不要写 “熟悉” zookeeper
整理了一些Java方面的架构.面试资料(微服务.集群.分布式.中间件等),有需要的小伙伴可以关注公众号[程序员内点事],无套路自行领取 一口气说出 9种 分布式ID生成方式,面试官有点懵了 面试总被问 ...
- OpenCV-Python 图像平滑 | 十六
目标 学会: 使用各种低通滤镜模糊图像 将定制的滤镜应用于图像(2D卷积) 2D卷积(图像过滤) 与一维信号一样,还可以使用各种低通滤波器(LPF),高通滤波器(HPF)等对图像进行滤波.LPF有助于 ...
- IntelliJ Idea 中文乱码问题
首先,Idea真的是一款很方便的开发工具,但是关于中文乱码这个问题我不得不吐槽,这个编码也弄得这么麻烦干嘛呀...下面就说一下怎么解决中文乱码问题: 1.首先是编辑器的乱码,这个很好解决,file-& ...
- Google浏览器截取整个网页
~Ctrl+shift+i(开发者工具) ~Ctrl+shift+p ~输入full
- 学习 MyBatis 的一点小总结 —— 底层源码初步分析
目录 MyBatis 如何获取数据库源? MyBatis 如何获取 sql 语句? MyBatis 如何执行 sql 语句? MyBatis 如何实现不同类型数据之间的转换? 在过去程序员使用 JDB ...
- java应用诊断和在线debug利器bistoury介绍与在K8S环境使用
Bistoury介绍 Bistoury 是去哪儿网开源的一个对应用透明,无侵入的java应用诊断工具,用于提升开发人员的诊断效率和能力,可以让开发人员无需登录机器或修改系统,就可以从日志.内存.线程. ...
- uCOS-II简介及移植uCOS-II到STM32F103平台详细步骤
1.参考博客:https://blog.csdn.net/wang328452854/article/details/78486458 2.uCOS(也有人叫uC/OS)由美国人 Jean Labro ...
- 好玩Python——PIL项目实训
PIL学习总结: 1. 2,PIL库概述: pil库可以完成图像归档和图像处理两方面功能的需求: 图像归档:对图像进行批处理,生成图像预览,图像转换格式等: 图像处理:图像基本处理,像素处理,颜色处理 ...