12号到今天共研究八数码问题poj1077,首先用的是普通BFS,遇到很多问题,开始用一个二级指针作为结构成员,知道了二级指针与二维数值名的不同!http://write.blog.csdn.net/postedit!讲得不错。发现自己代码能力渣死!进入正题,用BFS过不了,先学习了八数码问题有解条件,并扩展到N数码有解问题。做到奇数偶数剪枝,过不了,太慢,于是学习了HASH判重,学习了康托展开(排列与数字的一一对应关系),效率大大增加!但是别人这样过了都,我是过不了啊!于是去学双广!没怎么看明白,,,很想放弃了!于是去学习了A*算法!(启发式搜索),46ms,BFS是最普通的A*,那么,我用优先队列优化,设置启发函数,把优先级高的先出队!本质已经破坏了BFS的分层搜索,倒是有点DFS了,我名之为bdfs!,有方向性!这样找一组解很容易!不用遍历所有!

于是去切了骑士问题,先是8格的的,再是n格的,接下!

#include<iostream> //优先(小的优先)队列BFS+hash判重+奇数偶数剪枝
#include<queue>
#include<string>
#include<cstdio>
#include<set>
#include<cstring>
using namespace std;
string a;
int f[4][2]={0,1,0,-1,1,0,-1,0};//方向
bool state[362881];
int fac[10]={1,1,2,6,24,120,720,5040,40320,362880};
struct xy
{
string aa; //当前状态
int x; //空格的坐标
int y;
int abs_to_end; //和目标数相同的方块数目!
string s; // 字符串,保存当前移动情况
bool operator < (const xy & now) const //自定义离目标小的优先!
{
return now.abs_to_end>abs_to_end;
}
};
int get_abs(string aa) //只要第一次获得,其他改变修改
{
int res=0;
for(int i=1;i<=9;i++)
{
if(aa[i]!='9'&&aa[i]==('0'+i))res++;
}
return res;
} bool has_solution() //无解条件,奇偶剪枝!
{
int t[9];int count=0;
for(int i=0;i<3;i++)
for(int j=0;j<3;j++)
t[count++]=a[i*3+j]-'0';
count=0;
for(int i=0;i<9;i++)
for(int j=i+1;j<9;j++)
if(t[j]!=9&&t[i]!=9&&t[j]<t[i])count++;
if(count%2==0)return 1;
return 0;
}
string bfs(int ox,int oy) //
{
priority_queue<xy>q;
xy dian; dian.x=ox;dian.y=oy; //赋初值
dian.s=""; //首地址的赋值,二维数组必先从一维建立。用的还是同一个! for(int i=0;i<3;i++)
{
for(int j=0;j<3;j++)
{
dian.aa+=a[i*3+j];
}
}
dian.abs_to_end=get_abs(dian.aa);
q.push(dian);
if(dian.aa=="123456789")return dian.s;
while(!q.empty())
{
xy temp=q.top();
q.pop(); //取队首拓展
for(int i=0;i<4;i++)
{ // xy next=temp;这样只是完全赋值,指针指的是同一块地址!
xy next(temp);
int space=next.x*3+next.y;
next.x=next.x+f[i][0]; //四个方向拓展。
next.y=next.y+f[i][1];
if(next.x>=0&&next.x<3&&next.y>=0&&next.y<3)
{
if(i==0)
{
if(next.s.size()>0&&'l'==next.s[next.s.size()-1])continue;
next.s+="r";
if(next.aa[space+1]=='1'+space+1)next.abs_to_end--;
if(next.aa[space+1]=='1'+space)next.abs_to_end++;
next.aa[space]=next.aa[space+1];
next.aa[space+1]='9';
}
else if(i==1)
{
if(next.s.size()>0&&'r'==next.s[next.s.size()-1])continue;
next.s+="l";
if(next.aa[space-1]=='1'+space-1)next.abs_to_end--;
if(next.aa[space-1]=='1'+space)next.abs_to_end++;
next.aa[space]=next.aa[space-1];
next.aa[space-1]='9';
}
else if(i==2)
{
if(next.s.size()>0&&'u'==next.s[next.s.size()-1])continue;
next.s+="d";
if(next.aa[space+3]=='1'+space+3)next.abs_to_end--;
if(next.aa[space+3]=='1'+space)next.abs_to_end++;
next.aa[space]=next.aa[space+3];
next.aa[space+3]='9';
}
else if(i==3)
{
if(next.s.size()>0&&'d'==next.s[next.s.size()-1])continue;
next.s+="u";
if(next.aa[space-3]=='1'+space-3)next.abs_to_end--;
if(next.aa[space-3]=='1'+space)next.abs_to_end++;
next.aa[space]=next.aa[space-3];
next.aa[space-3]='9';
}
if(next.aa=="123456789")return next.s;
string ts=next.aa; //int hash(string ts)
int res=0;
for(int i=0;i<9;i++)
{
int count=0;
for(int j=i+1;j<9;j++)
if(ts[j]<ts[i])count++;
res+=count*fac[9-i-1];
}
if(state[res])continue;
state[res]=true;
q.push(next);
}
}
}
}
int main()
{
string ta;
while(getline(cin,ta))
{
int ox,oy;
a.clear();
for(int i=0;i<ta.size();i++)
{
if(ta[i]=='x') a+='9';
else if(ta[i]!=' ')a+=ta[i];
}
for(int i=0;i<a.size();i++)
{
if(a[i]=='9'){ox=i/3;oy=i%3;}
}
memset(state,0,sizeof(state));
if(!has_solution())cout<<"unsolvable"<<endl;
else
{
cout<<bfs(ox,oy)<<endl;
}
}
return 0;
}

由八数码问题引入。对BFS有更深考虑的更多相关文章

  1. 习题:八数码难题(双向BFS)

    八数码难题(wikioi1225) [题目描述] 在3×3的棋盘上,摆有八个棋子,每个棋子上标有1至8的某一数字.棋盘中留有一个空格,空格用0来表示.空格周围的棋子可以移到空格中.要求解的问题是:给出 ...

  2. HDU-1043 Eight八数码 搜索问题(bfs+hash 打表 IDA* 等)

    题目链接 https://vjudge.net/problem/HDU-1043 经典的八数码问题,学过算法的老哥都会拿它练搜索 题意: 给出每行一组的数据,每组数据代表3*3的八数码表,要求程序复原 ...

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

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

  4. Eight(South Central USA 1998)(八数码) 分类: bfs 2015-07-05 22:34 1人阅读 评论(0) 收藏

    The 15-puzzle has been around for over 100 years; even if you don't know it by that name, you've see ...

  5. 【洛谷P1379】八数码难题 状压bfs

    对于这道题来说,每个时刻的状态是整个棋盘所有棋子的位置,即:任何一个棋子位置发生了移动,都会使得状态转移. 因此,需要采取将整个状态作为广搜的搜索对象,进行状态压缩.采用哈希得到每个状态的对应的数值, ...

  6. 洛谷 P1379 八数码难题(map && 双向bfs)

    题目传送门 解题思路: 一道bfs,本题最难的一点就是如何储存已经被访问过的状态,如果直接开一个bool数组,空间肯定会炸,所以我们要用另一个数据结构存,STL大法好,用map来存,直接AC. AC代 ...

  7. BFS(八数码) POJ 1077 || HDOJ 1043 Eight

    题目传送门1 2 题意:从无序到有序移动的方案,即最后成1 2 3 4 5 6 7 8 0 分析:八数码经典问题.POJ是一次,HDOJ是多次.因为康托展开还不会,也写不了什么,HDOJ需要从最后的状 ...

  8. UVALive 6665 Dragon’s Cruller --BFS,类八数码问题

    题意大概就是八数码问题,只不过把空格的移动方式改变了:空格能够向前或向后移动一格或三格(循环的). 分析:其实跟八数码问题差不多,用康托展开记录状态,bfs即可. 代码: #include <i ...

  9. [cdoj1380] Xiper的奇妙历险(3) (八数码问题 bfs + 预处理)

    快要NOIP 2016 了,现在已经停课集训了.计划用10天来复习以前学习过的所有内容.首先就是搜索. 八数码是一道很经典的搜索题,普通的bfs就可求出.为了优化效率,我曾经用过康托展开来优化空间,甚 ...

随机推荐

  1. TigerGraph REST++API

    简介 - 什么是REST ++? TigerGraph TM 系统使用着名的REpresentational State Transfer(REST)架构来管理与TigerGraph核心组件,图形处理 ...

  2. redis集群架构(含面试题解析)

    老规矩,我还是以循序渐进的方式来讲,我一共经历过三套集群架构的演进! Replication+Sentinel 这套架构使用的是社区版本推出的原生高可用解决方案,其架构图如下! 这里Sentinel的 ...

  3. 设置QtreeWidget水平滚动条

    转载请注明出处:http://www.cnblogs.com/dachen408/p/7552603.html //设置treewidget水平滚动条 ui.treeWidget->header ...

  4. 接口自动化- 基于 Python

    准备工作 这部分其实在谷歌或者百度上搜索下就可以完成的,可是我就是想再啰嗦一遍,说不定有比我更懒的同学呢哈哈~ 第一步 Python的安装配置 打开官网: https://www.python.org ...

  5. codevs 1979 第K个数

     时间限制: 1 s  空间限制: 1000 KB  题目等级 : 黄金 Gold 题目描述 Description 给定一个长度为N(0<n<=10000)的序列,保证每一个序列中的数字 ...

  6. MFC技术积累——基于MFC对话框类的那些事儿5

    4. 菜单 4.1 弹出菜单 本节主要讲解如何在主对话框的指定区域内通过鼠标右击来弹出一个菜单选项.最终效果图如图4.1. 如图4.1鼠标只能在指定区域(图中深色区域)内右击时弹出菜单,在指定区域外点 ...

  7. Microsoft Project 2010基础使用方法

    5.1 项目管理与Microsoft Project2010 Microsoft Project2010深受广大项目管理工程师的青睐. 5.1.1 项目管理的概念 项目管理是项目管理者在有限的资源约束 ...

  8. k8s集群部署之环境介绍与etcd数据库集群部署

    角色 IP 组件 配置 master-1 192.168.10.11 kube-apiserver kube-controller-manager kube-scheduler etcd 2c 2g ...

  9. java中等待所有线程都执行结束

    转自:http://blog.csdn.net/liweisnake/article/details/12966761 今天看到一篇文章,是关于java中如何等待所有线程都执行结束,文章总结得很好,原 ...

  10. 前端开发中的 meta 整理

    meta是html语言head区的一个辅助性标签.也许你认为这些代码可有可无.其实如果你能够用好meta标签,会给你带来意想不到的效果,meta标签的作用有:搜索引擎优化(SEO),定义页面使用语言, ...