<题目链接>

题目大意:
给出一个3×3的矩阵(包含1~8数字和一个字母x),经过一些移动格子上的数后得到连续的1~8,最后一格是x,要求最小移动步数。

解题分析:
本题用BFS来寻找路径,为了降低复杂度,用BFS从最终的目标状态开始处理,将所有搜索到状态以及对应的路径打表记录,然后对于输入的矩阵,直接查表输出答案 即可,本题还有一个难点,就是如何判断记录已经搜索过的状态,如果使用map+string(矩阵看成一维)会超时,所以我们这里用康拓展开式来记录状态。这道题比较玄学的地方在于如果更换dir 中四个方向的顺序,可能会WA,而在题目中没有对输出的字符串做任何限制,不知道这是为什么。

#include<cstdio>
#include<iostream>
#include<queue>
using namespace std;
typedef struct nn
{
char way; //记录操作
int fath; //记录父节点,用于记录路径
}node1; typedef struct nod
{
int aa[];
int n; //n为9在aa中的位置
int son; //记录aa的康拓展开式
}node2; int dir[][]={{,},{,-},{,},{-,}},fac[];
node1 Node[];//节点 void set_fac()//计算0到8的阶层
{
fac[]=;
for(int i=;i<=;i++)
fac[i]=fac[i-]*i;
} int cantor(int aa[])//康托展开,掌握康拓展开的方法
{
int ans=;
for(int i=;i<;i++)
{
int k=;
for(int j=i+;j<;j++)
if(aa[i]>aa[j])
k++;
ans+=k*fac[-i]; //i点以后比aa[i]小的数的个数*((n-i)-1)! 之和
}
return ans;
} void bfs(int a[])
{
queue<node2>Q;
node2 now,next; for(int i=;i<;i++) now.aa[i]=a[i];
now.n=;now.son=;
Node[now.son].fath=; //把最终父节点记为0,也就是本身
Q.push(now);
while(!Q.empty())
{
now=Q.front(); Q.pop();
for(int k=;k<;k++)
{
next=now;
int tx=now.n/+dir[k][];
int ty=now.n%+dir[k][];
if(ty>=&&tx>=&&ty<&&tx<)
{
next.n=tx*+ty; //得到移动后x的位置
int tem=next.aa[next.n]; //得到移动后该位置的原数字
next.aa[next.n]=next.aa[now.n];
next.aa[now.n]=tem; //将两个位置上的数字交换
next.son=cantor(next.aa); if(Node[next.son].fath==-) //为-1时表示这个点没有访问过,那么放入队列
{
Node[next.son].fath=now.son; //当前节点的父节点就是上一个节点
if(k==)Node[next.son].way='l';//一定要注意了,k=0是向右走,但我们是从终止状态往回搜,所以直接记录相反的方向
if(k==)Node[next.son].way='r';
if(k==)Node[next.son].way='u';
if(k==)Node[next.son].way='d';
Q.push(next);
}
}
}
}
} int main()
{
int i,j,s,ss[],a[]; for(i=;i<;i++)//目标状态
a[i]=i+; //建立目标一维矩阵,把x看成9 for(i=;i<;i++)
Node[i].fath=-;
set_fac(); //计算阶层
bfs(a); //将从最终状态能够延伸出去的所有状态以及对应路径提前打表记录 char str[];
while(gets(str))
{
for(i=,j=;str[i]!='\0';i++)//把字符串变成数子
{
if(str[i]=='x')
ss[j++]=; //把x变为数子9
else if(str[i]>=''&&str[i]<='')
ss[j++]=str[i]-'';
}
s=cantor(ss); //算出初态康托值
if(Node[s].fath==-) {printf("unsolvable\n");continue;} //不能变成目标,因为当从起点开始搜索的时候,fath表示搜索到某
//点时,上一个点的状态,所以,如果s.fath==-1时,表示这个矩阵根本就延伸不出去,不可能达到目标状态 while(s!=)
{
printf("%c",Node[s].way);
s=Node[s].fath;
}
printf("\n");
}
}

2018-09-06

hdu 1043 Eight (八数码问题)【BFS】+【康拓展开】的更多相关文章

  1. hdu 1043 pku poj 1077 Eight (BFS + 康拓展开)

    http://acm.hdu.edu.cn/showproblem.php?pid=1043 http://poj.org/problem?id=1077 Eight Time Limit: 1000 ...

  2. HDU 1043 Eight 八数码问题 A*算法(经典问题)

    HDU 1043 Eight 八数码问题(经典问题) 题意 经典问题,就不再进行解释了. 这里主要是给你一个状态,然后要你求其到达\(1,2,3,4,5,6,7,8,x\)的转移路径. 解题思路 这里 ...

  3. Hdu 1043 Eight (八数码问题)

    题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=1043 题目描述: 3*3的格子,填有1到8,8个数字,还有一个x,x可以上下左右移动,问最终能否移动 ...

  4. 【HDU - 1043】Eight(反向bfs+康托展开)

    Eight Descriptions: 简单介绍一下八数码问题:在一个3×3的九宫格上,填有1~8八个数字,空余一个位置,例如下图: 1 2 3 4 5 6 7 8   在上图中,由于右下角位置是空的 ...

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

  6. HUD 1043 Eight 八数码问题 A*算法 1667 The Rotation Game IDA*算法

    先是这周是搜索的题,网站:http://acm.hdu.edu.cn/webcontest/contest_show.php?cid=6041 主要内容是BFS,A*,IDA*,还有一道K短路的,.. ...

  7. HDU 4531 bfs/康拓展开

    题目链接http://acm.hdu.edu.cn/showproblem.php?pid=4531 吉哥系列故事——乾坤大挪移 Time Limit: 2000/1000 MS (Java/Othe ...

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

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

  9. poj 1077-Eight(八数码+逆向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. Mysql8.0安装步骤

    Mysql8.0安装步骤 2018年05月10日 14:39:05 93年的香槟 阅读数:19628 标签: mysql 更多 个人分类: 数据库   版权声明:本文为博主原创文章,未经博主允许不得转 ...

  2. Confluence 6 使用 Apache 和 mod_proxy 的基本配置

    在这些示例中,我们使用下面的信息: http://www.example.com/confluence - 你计划使用的 URL http://example:8090/ - Confluence 当 ...

  3. Confluence 6 配置校验和识别

    校验你的设置 查看你 Confluence 当前使用的设置,请参考 Viewing System Properties 页面中的内容. 识别系统属性 请参考 Recognized System Pro ...

  4. Confluence 6 针对你的数据库类型确定校验 SQL

    不同的数据库通常要求不同的 SQL 校验查询.校验查询通常需要尽可能的简单,这个查询在链接从数据库连接池中取出的时候都会被执行一次. 针对不同的数据库类型,我们推荐先的校验查询 SQL: MySQL ...

  5. RemoveDuplicatesfromSortedList

    给定一个排序链表,删除所有重复的元素,使得每个元素只出现一次. 示例 1: 输入: 1->1->2 输出: 1->2 示例 2: 输入: 1->1->2->3-&g ...

  6. 利用Form组件和ajax实现的注册

    一.注册相关的知识点 1.Form组件 我们一般写Form的时候都是把它写在views视图里面,那么他和我们的视图函数也不影响,我们可以吧它单另拿出来,在应用下面建一个forms.py的文件来存放 2 ...

  7. cf1114D 区间dp基础

    最简单的那类区间dp,昨天晚上心态不对,不知道在打什么.. /* dp[l][r]表示区间[l,r]都涂成同色的代价 dp[l][r]可以由dp[l][r-1],dp[l+1][r],dp[l+1][ ...

  8. npm 如何查看一个包的版本信息?

    转载. https://blog.csdn.net/cvper/article/details/79543262 有了npm 我们能够简单的一段代码就下载我们需要的包,但是包是不断更新的, 所以我们要 ...

  9. C++ Primer 笔记——枚举类型

    1.和类一样,每个枚举类型定义了一种新的类型.枚举属于字面值常量类型. 2.C++包含两种枚举:限定作用域的和不限定作用域的.C++11新标准引入了限定作用域的枚举类型. }; // 限定作用域的枚举 ...

  10. C++ Primer 笔记——智能指针

    1.新的标准库提供了两种智能指针类型,shared_ptr允许多个指针指向同一个对象,unique_ptr则独占所指的对象.标准库还定义了一个名为weak_ptr的伴随类,它是一种弱引用,指向shar ...