题意:就是说,给出一个三行三列的数组,其中元素为1--8和x,例如:

1  2  3    现在,需要你把它变成:1  2  3      要的最少步数的移动方案。可以右移r,左移l,上移u,下移d
x 4 6 4 5 6
7 5 8 7 8 x

思路:这是赤裸裸的康托展开了吧?不知道康托展开的同学可以百度百科........好吧,其实我想说的是,这个题目怎么分析出,要用到康托展开的。首先,遇到这样的题目,元素比较少,又可以用到搜索,但是状态数比较多,也就是说,我需要找到一种方式来压缩这些状态,使得同一个状态不重复出现。额,说道这里,在不考虑时间、空间复杂度的情况下,的确可以直接开标记数组:vist[10][10][10][10][10][10][10][10][10],但是这样一来,会有许多重复的状态。有没有更好压缩状态的方法呢?是有的,进制压缩也是一种方法,只是对于这个题目不适用。那么还有康托展开,康托展开,可以使得这些状态压缩下来,只有9!个状态,如此就可以实现搜索了。

有的人说,广搜不能记录路径?这是错误的,广搜只要你清楚的记录这个状态的上个状态是什么,那么就可以记录路径,很明显只需要开个记录路径的搜索就ok了

代码:

#include<iostream>
#include<cstdio>
#include<cstring>
#include<queue>
using namespace std;
int t[10]= {1,1,2,6,24,120,720,5040,40320};
char ff[400000];
struct node
{
char ch[10];
int son;
int wz;
};
struct node1 //记录路径
{
char num;
int father;
} s[362883]; int deal(char str[]) //康托展开
{
int x[10],ans1=0;
for(int i=0; i<9; i++)
x[i]=str[i]-'0'; /*for(int i=0;i<9;i++)
printf("%d\t",x[i]);*/ for(int i=0; i<9; i++)
{
int k=0;
for(int j=i+1; j<9; j++)
if(x[i]>x[j])
k++;
ans1+=k*t[8-i];
//printf("%d %d %d\n",ans1,k,t[8-i]);
}
return ans1;
} int flag=0;
void bfs(char str[])
{
queue<node>q;
node p;
strcpy(p.ch,str);
p.ch[9]='\0';
p.son=deal(p.ch);
// puts(p.ch);
p.wz=8;
//printf("%d\n",p.son);
s[p.son].father=0; q.push(p); while(!q.empty())
{
//flag++;
//printf("%d\n",flag);
p=q.front();
q.pop();
int hang=p.wz/3;
int lie=p.wz%3;
{
int x=hang-1;
int y=lie;
{
int x=hang;
int y=lie+1;
if(x>=0&&x<3&&y>=0&&y<3&&(x*3+y)<=8)
{
node p1;
strcpy(p1.ch,p.ch);
p1.wz=x*3+y;
char zf=p1.ch[p.wz];
p1.ch[p.wz]=p1.ch[p1.wz];
p1.ch[p1.wz]=zf;
p1.ch[9]='\0';
p1.son=deal(p1.ch);
if(s[p1.son].father==-1)
{
s[p1.son].father=p.son;
s[p1.son].num='l';
q.push(p1);
}
}
} {
int x=hang;
int y=lie-1;
if(x>=0&&x<3&&y>=0&&y<3&&(x*3+y)<=8)
{
node p1;
strcpy(p1.ch,p.ch);
p1.wz=x*3+y;
char zf=p1.ch[p.wz];
p1.ch[p.wz]=p1.ch[p1.wz];
p1.ch[p1.wz]=zf;
p1.ch[9]='\0';
p1.son=deal(p1.ch);
if(s[p1.son].father==-1)
{
s[p1.son].father=p.son;
s[p1.son].num='r';
q.push(p1);
}
}
}
if(x>=0&&x<3&&y>=0&&y<3&&(x*3+y)<=8)
{
node p1;
strcpy(p1.ch,p.ch);
p1.wz=x*3+y;
char zf=p1.ch[p.wz];
p1.ch[p.wz]=p1.ch[p1.wz];
p1.ch[p1.wz]=zf;
p1.ch[9]='\0';
p1.son=deal(p1.ch);
if(s[p1.son].father==-1)
{
s[p1.son].father=p.son;
s[p1.son].num='d';
q.push(p1);
}
}
} {
int x=hang+1;
int y=lie;
if(x>=0&&x<3&&y>=0&&y<3&&(x*3+y)<=8)
{
node p1;
strcpy(p1.ch,p.ch);
p1.wz=x*3+y;
char zf=p1.ch[p.wz];
p1.ch[p.wz]=p1.ch[p1.wz];
p1.ch[p1.wz]=zf;
p1.ch[9]='\0';
p1.son=deal(p1.ch);
if(s[p1.son].father==-1)
{
s[p1.son].father=p.son;
s[p1.son].num='u';
q.push(p1);
}
}
} }
}
int main()
{
char str[10]= {'1','2','3','4','5','6','7','8','0'};
for(int i=0; i<362883; i++)
{
s[i].father=-1;
}
//printf("%d\n",flag);
bfs(str);
char ss[10][10];
char tt[10];
scanf("%s",ss[0]);
{
if(ss[0][0]!='x')
tt[0]=ss[0][0];
else
tt[0]='0'; //把x转化为数字0
for(int i=1; i<9; i++)
{
scanf("%s",ss[i]);
if(ss[i][0]!='x')
tt[i]=ss[i][0];
else
tt[i]='0';
}
tt[9]='\0';
//puts(tt);
int ans=deal(tt);
int cnt=0;
flag=0;
while(ans!=0) //回溯路径
{
ff[cnt++]=s[ans].num;
ans=s[ans].father;
if(ans==-1)
{
flag=1;
break;
}
}
int i=0;
if(flag==0)
{
while(cnt>i)
printf("%c",ff[i++]);
}
else printf("unsolvable");
printf("\n");
}
return 0;
}

poj1077(康托展开+bfs+记忆路径)的更多相关文章

  1. hdu 1430(BFS+康托展开+映射+输出路径)

    魔板 Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Submiss ...

  2. hdu1043Eight (经典的八数码)(康托展开+BFS)

    建议先学会用康托展开:http://blog.csdn.net/u010372095/article/details/9904497 Problem Description The 15-puzzle ...

  3. HDU 1430 魔板(康托展开+BFS+预处理)

    魔板 Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total Submis ...

  4. HDU 1043 & POJ 1077 Eight(康托展开+BFS+预处理)

    Eight Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 30176   Accepted: 13119   Special ...

  5. HDU 3567 Eight II 打表,康托展开,bfs,g++提交可过c++不可过 难度:3

    http://acm.hdu.edu.cn/showproblem.php?pid=3567 相比Eight,似乎只是把目标状态由确定的改成不确定的,但是康托展开+曼哈顿为h值的A*和IDA*都不过, ...

  6. [HDOJ1043]Eight(康托展开 BFS 打表)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1043 八数码问题,因为固定了位置所以以目标位置开始搜索,把所有情况(相当于一个排列)都记录下来,用康托 ...

  7. HDU 1043 & POJ 1077 Eight(康托展开+BFS | IDA*)

    Eight Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 30176   Accepted: 13119   Special ...

  8. poj2965(位运算压缩+bfs+记忆路径)

    题意:有个4*4的开关,里面有着16个小开关 -+-- ---- ---- '+'表示开关是关着的,'-'表示开关是开着的,只有所有的开关全被打开,总开关才会被打开.现在有一种操作,只要改变某个开关, ...

  9. HDU 1043 Eight 【经典八数码输出路径/BFS/A*/康托展开】

    本题有写法好几个写法,但主要思路是BFS: No.1 采用双向宽搜,分别从起始态和结束态进行宽搜,暴力判重.如果只进行单向会超时. No.2 采用hash进行判重,宽搜采用单向就可以AC. No.3 ...

随机推荐

  1. LC_ALL=C的含义

    在很多的shell脚本中,我们经常会看见某一句命令的前面有一句“LC_ALL=C” SAR_CMD="LC_ALL=C sar -u -b 1 5 | grep -i average &qu ...

  2. 31天重构学习笔记(java版本)

    准备下周分享会的内容,无意间看到.net版本的重构31天,花了两个小时看了下,可以看成是Martin Fowler<重构>的精简版 原文地址:http://www.lostechies.c ...

  3. Linux系统排查——CPU负载篇

    本随笔介绍CPU负载的排查手段. 查看系统负载的工具:uptime,w,都能查看系统负载,系统平均负载是处于运行或不可打扰状态的进程的平均数, 可运行:运行态,占用CPU,或就绪态,等待CPU调度. ...

  4. Activity的onPause()、onStop()和onDestroy()里要做的事情

    onPause(): 当系统调用你的activity中的onPause(),从技术上讲,那意味着你的activity仍然处于部分可见的状态,当时大多数时候,那意味着用户正在离开这个activity并马 ...

  5. 带你开发一款给Apk中自己主动注入代码工具icodetools(开凿篇)

    一.前言 从这篇開始咋们開始一个全新的静态方式逆向工具icodetools的实现过程.这个也是我自己第一次写的个人认为比較实用的小工具,特别是在静态方式逆向apk找关键点的时候.兴许会分为三篇来具体介 ...

  6. C++编程技巧降低编译时间

    1. #define的保护 全部头文件都应该使用#define 防止头文件被多重包括(multiple inclusion).命名格式 当是:<PROJECT>_<PATH>_ ...

  7. mongodb及mongoclient在win7下的编译和使用

    1.导言 折腾mongodb几个小时最终有结果了.呃.如今就简单总结一下. 事实上我的需求非常easy,就是在C++代码中调用mongodb的库函数.也就是要得到mongoclient.lib.本来想 ...

  8. Python selenium 滚动条 详解

    在我们使用Python + selenium 爬虫的时候,会遇到如下报错,原因是  当页面上的元素超过一屏后,想操作屏幕下方的元素,是不能直接定位到,会报元素不可见的. selenium.common ...

  9. python标准库介绍——37 signal 模块详解

    ==signal 模块== 你可以使用 ``signal`` 模块配置你自己的信号处理器 (signal handler), 如 [Example 3-11 #eg-3-11] 所示. 当解释器收到某 ...

  10. 使用 http 请求方式获取 eureka server的服务信息

    对于一些系统不能接入 eureka server,又需要访问接入eureka server 的服务. 方法一:直接调用服务的地址是一种实现方式,弊端就是地址是写死的,万一服务地址变更则访问不到. 方法 ...