题意:经典八数码问题

思路:双向bfs

ps:还有a*算法(还不会)等解法。

代码:

#include<iostream>
#include<stdio.h>
#include<string.h>
#include<queue>
using namespace std; #define MAXN 362885//最多组合个数:9!=362880 int dir[]={-,,-,};//4个方向移动:u,d,l,r
char opF[]={'u','d','l','r'};//正向bfs操作
char opR[]={'d','u','r','l'};//反向bfs操作
int fac[]={,,,,,,,,};//阶乘
bool visF[MAXN];//正向访问标志
bool visR[MAXN];//
string sTarg="";//目标棋盘 struct node{//存储棋盘信息
string s;//当前棋盘
int xLoca;//x位置
}t; struct node2{//存储操作
int idF,idR;//正向前驱、反向前驱
char cF,cR;//操作
}op[MAXN]; int cantor_hash(string s){//康托展开,一共n位
int n=;//共9位
int i,j,temp,num;
num=;
for(i=;i<n-;i++){//n为位数
temp=;
for(j=i+;j<n;j++){
if(s[j]<s[i])temp++;
}
num+=fac[n-(i+)]*temp;
}
return num;//从0开始
} void display(int i){//输出正向操作
if(op[i].idF==-)return;
display(op[i].idF);
printf("%c",op[i].cF);
} void dBfs(){
queue<node>qF,qR;//正向队列,反向队列
node tmp1,tmp2;//棋盘状态临时变量
int id_hash,id_hash2,xLoca,i;//id 康托哈希下标值,xLoca 暂存x位置 tmp1=t;//初始状态 tmp2.s=sTarg;//目标状态
tmp2.xLoca=;//目标状态x位置 id_hash=cantor_hash(tmp1.s);//初始状态
visF[id_hash]=true;
op[id_hash].idF=-;
qF.push(tmp1); id_hash=cantor_hash(tmp2.s);//目标状态
visR[id_hash]=true;
op[id_hash].idR=-;
qR.push(tmp2); while(!qF.empty()&&!qR.empty()){
//正向bfs
tmp1=qF.front();
qF.pop();
id_hash=cantor_hash(tmp1.s);
if(visR[id_hash]){//反向bfs也访问过
display(id_hash);
id_hash2=id_hash;
while(op[id_hash2].idR!=-){
printf("%c",op[id_hash2].cR);
id_hash2=op[id_hash2].idR;
}
printf("\n");
return;
}
for(i=;i<;++i){
if(i==&&tmp1.xLoca<)continue;
if(i==&&tmp1.xLoca>)continue;
if(i==&&tmp1.xLoca%==)continue;
if(i==&&tmp1.xLoca%==)continue;
xLoca=tmp1.xLoca+dir[i]; tmp2=tmp1;
swap(tmp2.s[tmp1.xLoca],tmp2.s[xLoca]);
id_hash2=cantor_hash(tmp2.s);
if(!visF[id_hash2]){
visF[id_hash2]=true;
tmp2.xLoca=xLoca;
op[id_hash2].idF=id_hash;
op[id_hash2].cF=opF[i];
qF.push(tmp2);
}
} //反向bfs
tmp1=qR.front();
qR.pop();
id_hash=cantor_hash(tmp1.s);
if(visF[id_hash]){//正向bfs也访问过
display(id_hash);
id_hash2=id_hash;
while(op[id_hash2].idR!=-){
printf("%c",op[id_hash2].cR);
id_hash2=op[id_hash2].idR;
}
printf("\n");
return;
}
for(i=;i<;++i){
if(i==&&tmp1.xLoca<)continue;
if(i==&&tmp1.xLoca>)continue;
if(i==&&tmp1.xLoca%==)continue;
if(i==&&tmp1.xLoca%==)continue;
xLoca=tmp1.xLoca+dir[i]; tmp2=tmp1;
swap(tmp2.s[tmp1.xLoca],tmp2.s[xLoca]);
id_hash2=cantor_hash(tmp2.s);
if(!visR[id_hash2]){
visR[id_hash2]=true;
tmp2.xLoca=xLoca;
op[id_hash2].idR=id_hash;
op[id_hash2].cR=opR[i];
qR.push(tmp2);
}
}
}
} int main(){
char str[];//串
int len;//串长度
int i,j,k;//k 第几个数码
int ivsNum;//逆序数
while(~scanf("%[^\n]",str)){
getchar();//吸收回车 memset(visF,false,sizeof(visF));
memset(visR,false,sizeof(visR)); len=strlen(str);
t.s="";
k=;//第几个数码
for(i=;i<len;++i){//处理字符串
if(str[i]!=' '){
if(str[i]=='x'){
t.s=t.s+'';
t.xLoca=k;
}
else t.s=t.s+str[i];
++k;//
}
} ivsNum=;//逆序数初始为0
for(i=;i<;++i){
if(t.s[i]=='')continue;
for(j=;j<i;++j){
if(t.s[j]=='')continue;
if(t.s[j]>t.s[i])++ivsNum;
}
}
if(ivsNum&)printf("unsolvable\n");//逆序数为奇数
else dBfs();//双向bfs
}
return ;
}

hdu 1043 Eight(双向bfs)的更多相关文章

  1. 2017多校第10场 HDU 6171 Admiral 双向BFS或者A*搜索

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6171 题意: 给你一个高度为6的塔形数组,你每次只能将0与他上下相邻的某个数交换,问最少交换多少次可以 ...

  2. HDU 1242 -Rescue (双向BFS)&amp;&amp;( BFS+优先队列)

    题目链接:Rescue 进度落下的太多了,哎╮(╯▽╰)╭,渣渣我总是埋怨进度比别人慢...为什么不试着改变一下捏.... 開始以为是水题,想敲一下练手的,后来发现并非一个简单的搜索题,BFS做肯定出 ...

  3. HDU 3085 Nightmare Ⅱ (双向BFS)

    Nightmare Ⅱ Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Tota ...

  4. HDU 3085 Nightmare Ⅱ 双向BFS

    题意:很好理解,然后注意几点,男的可以一秒走三步,也就是三步以内的都可以,鬼可以穿墙,但是人不可以,鬼是一次走两步 分析:我刚开始男女,鬼BFS三遍,然后最后处理答案,严重超时,然后上网看题解,发现是 ...

  5. HDU 1043 Eight(反向BFS+打表+康托展开)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1043 题目大意:传统八数码问题 解题思路:就是从“12345678x”这个终点状态开始反向BFS,将各 ...

  6. Nightmare Ⅱ HDU - 3085 (双向bfs)

    Last night, little erriyue had a horrible nightmare. He dreamed that he and his girl friend were tra ...

  7. ACM-BFS之Open the Lock——hdu1195(双向BFS)

    这道题的0基础版本号,暴力BFS及题目详情请戳:http://blog.csdn.net/lttree/article/details/24658031 上回书说道,要用双向BFS来尝试一下. 最终A ...

  8. HDU 1043 Eight(双向BFS+康托展开)

    http://acm.hdu.edu.cn/showproblem.php?pid=1043 题意:给出一个八数码,求出到达指定状态的路径. 思路:路径寻找问题.在这道题里用到的知识点挺多的.第一次用 ...

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

随机推荐

  1. POJ 3461 kmp 应用

    题意:求匹配串在文本中出现次数,KMP应用,理解了就OK了,每次匹配成功就累加次数,开始的时候超时, 由于在处理每次成功的时候让i=i-len2+1,相当于回溯了,后来一想,本次成功,相当于" ...

  2. solus系统配置

    #更新软件源 清华稳定源 sudo eopkg ar Tuna https://mirrors.tuna.tsinghua.edu.cn/solus/shannon/eopkg-index.xml 清 ...

  3. HDU 5905 Black White Tree(树型DP)

    题目链接  Black White Tree 树型DP,设$f[i][j]$为以$i$为根的子树中大小为$j$的连通块中可以包含的最小黑点数目. $g[i][j]$为以$i$为根的子树中大小为$j$的 ...

  4. java基础 6 基本类型与运算

    1 包装类型 Integer的缓存池为   -128 - 127: 八个基本类型   占bit  与字节  8 bit = 1 字节 boolean  1     byte  8    char   ...

  5. HTML小知识点积累

    1.怎样让heigth:100%起效?   有时候我们设置heigth:100%,想让当前控件铺满整个屏幕,可是非常少情况下这个属性能达到我们想要的效果,这是为什么呢?   而依据W3C的规范.百分比 ...

  6. Git的使用 -- 用git玩翻github,结尾有惊喜!有惊喜!有惊喜!林妙妙看了说:牛呲呼啦带闪电 (三)(超详解)

    简介 上一篇主要讲解的是Git安装及配置,这一篇就详细的从无到有的来用Git玩翻github. 一.什么是Github Github是全球最大的社交编程及代码托管网站(https://github.c ...

  7. 微软开源 Try .NET - 创建交互式.NET文档

    微软近日开源了一个新平台--Try .NET,该平台可以让开发者在线上编写并运行 .NET 代码.微软介绍,Try .NET 是一个可嵌入的代码运行器,不仅可以直接在线上对自己或者他人的代码进行编辑. ...

  8. LeetCode(28)题解:Implement strStr()

    https://leetcode.com/problems/implement-strstr/ 题目: Implement strStr(). Returns the index of the fir ...

  9. 在vc6.0下编的对话框界面如果没做过其他处理,往往显的很生硬,怎么样才能使他有Windows XP的风格呢,其实也很简单,我们来看看下面两种方法。

    在vc6.0下编的对话框界面如果没做过其他处理,往往显的很生硬,怎么样才能使他有Windows XP的风格呢,其实也很简单,我们来看看下面两种方法.    方法一: 1.首先确认你在Windows   ...

  10. javascript模块化编程:CommonJS和AMD规范

    AMD规范,异步模块定义.与CommonJS规范齐名并列. 作用都是利于JavaScript的模块化编程. 模块化编程的好处就是: 1.可重用 2.独立 3.能解决加载的依赖性问题 4.能解决重复加载 ...