在一个3*3的棋盘上放置编号为1~8的八个方块,每个占一格,另外还有一个空格。与空格相邻的数字方块可以移动到空格里。任务1:指定的初始棋局和目标棋局,计算出最少的移动步数;任务2:数出数码的移动序列。

  把空格看成0,一共有九个数字。

  输入样例:

  1 2 3 0 8 4 7 6 5

  1 0 3 8 2 3 7 6 5

  输出样例:

  2

  1.把一个棋局看成一个状态图,总共有9!= 362880个状态。从初始棋局开始,每次移动转到下一个状态,达到目标棋局后停止。

  2.康托展开

    康托展开是一种特殊的哈希函数。其功能是在输入一个排列,计算出它在在全排列中从小到大排序的位次。

    eg:判断 2143是{1,2,3,4}的全排列中的位次。

      (1)首位小于2的所有排列。比2小的只有1,后面三个数的排列有3*2*1=3!个,写成1*3!=6

      (2)首位为2,第二位小于1的所有排列。无,写成0*2!=0

      (3)前两位为21,第三位小于4的所有排列。只有3一个数,写成1*1!=1

      (3)前三位为214,第四位小于3的所有排列。无,写成0*0!=0

      求和:1*3!+0*2!+1*1!+0*0!=7

    所以位次的计算公式为X = a[n]*(n-1)! +a[n-1]*(n-2)! + … + a[1]*0!

  

 #include<bits/stdc++.h>
#include<queue>
using namespace std; const int len = ; //状态共9! = 362880种
int visited[len] = {};//标记已有状态用来去重
int start[];//起始状态
int goal[];//目标状态
int factory[] = {, , , , , , , , , };//0到9的阶乘
int dir[][] = {{-, }, {, -}, {, }, {, }}; struct node{
int state[];//棋局状态按一维存放下来
int dis;//记录从起始状态移动到当前状态的步数
}; bool cantor(int str[], int n){
int result = ;
for(int i=; i<n; i++){
int cnt = ;
for(int j=i+; j<n; j++){
if(str[i] > str[j])
cnt ++;
}
result += cnt*factory[n-i-];
}
if(!visited[result]){
visited[result] = ;
return ;
}
return ;
} int BFS(){
node head, next;
memcpy(head.state, start, sizeof(head.state));//复制起始状态并插入队列
head.dis = ;
cantor(head.state, );
queue<node>q;
q.push(head); while(!q.empty()){
head = q.front();
q.pop();
int z;
for(z=; z<; z++)
if(head.state[z] == )//找到0
break;
int x = z % ;//将0的一维位置转化为二维的横纵坐标
int y = z / ;
for(int i=; i<; i++){
int newx = x + dir[i][];
int newy = y + dir[i][];
int newz = newx + *newy;//将0移动后重新转化为一维坐标
if(newx>= && newx< && newy>= && newy<){//避免越界
memcpy(&next, &head, sizeof(struct node));
swap(next.state[z], next.state[newz]);//复制原先状态后,改变0的位置
next.dis ++;
if(memcmp(next.state, goal, sizeof(next.state)) == )
return next.dis;
if(cantor(next.state, ))//查重
q.push(next);
}
}
}
return -;
} int main(){
for(int i=; i<; i++)
scanf("%d", start+i);
for(int i=; i<; i++)
scanf("%d", goal+i); int num = BFS();
if(num != -)
printf("%d\n",num);
else
printf("Impossible\n");
}

  (1)用于存放状态图的以及步数的结构体

  (2)用于移动的数组

  (3)用于去重的标记数组

  (4)提前算好阶乘存放于数组中

  (5)康拓函数判重

  (6)BFS函数:queue<node>q;node head, next;

  (7)状态图中某数字方块的一维坐标和二维坐标的相互转化

  (8)检查坐标是否合法

八数码问题多种解法:https://www.cnblogs.com/zufezzt/p/5659276.html

BFS-八数码问题与状态图搜索的更多相关文章

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

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

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

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

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

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

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

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

  5. 八数码问题+路径寻找问题+bfs(隐式图的判重操作)

    Δ路径寻找问题可以归结为隐式图的遍历,它的任务是找到一条凑够初始状态到终止问题的最优路径, 而不是像回溯法那样找到一个符合某些要求的解. 八数码问题就是路径查找问题背景下的经典训练题目. 程序框架 p ...

  6. HDU 1043 Eight (BFS&#183;八数码&#183;康托展开)

    题意  输出八数码问题从给定状态到12345678x的路径 用康托展开将排列相应为整数  即这个排列在全部排列中的字典序  然后就是基础的BFS了 #include <bits/stdc++.h ...

  7. 【算法】BFS+哈希解决八数码问题

    15拼图已经有超过100年; 即使你不叫这个名字知道的话,你已经看到了.它被构造成具有15滑动砖,每一个从1到15上,并且所有包装成4乘4帧与一个瓦块丢失.让我们把丢失的瓷砖“X”; 拼图的目的是安排 ...

  8. hdu-1043(八数码+bfs打表+康托展开)

    参考文章:https://www.cnblogs.com/Inkblots/p/4846948.html 康托展开:https://blog.csdn.net/wbin233/article/deta ...

  9. 关于八数码问题中的状态判重的三种解决方法(编码、hash、&lt;set&gt;)

    八数码问题搜索有非常多高效方法:如A*算法.双向广搜等 但在搜索过程中都会遇到同一个问题.那就是判重操作(假设反复就剪枝),怎样高效的判重是8数码问题中效率的关键 以下关于几种判重方法进行比較:编码. ...

随机推荐

  1. codeforces 995C

    题意:从L到R 找有几个x,使x=a^p(a>0,p>1) 题解: 一开始把所有符合的次方都存到vector,然后MLE 可以看到1e6^3=1e18,所以可以将二次方单独来求,其他次方存 ...

  2. 剑指offer-面试题45-把数组排成最小的数-规律

    /* 题目: 给定一个int数组,返回数组中各数字排成的最下字符串. */ /* 思路: 比较两个数字之间的先后顺序,谁排在前面更小,从而对数组进行排序,得到结果. 两个数字先后顺序的比较方法:两个数 ...

  3. STL入门学习中碰到的一些函数

    2020.02.10 fill #include<algorithm> vector<int> v{ 1, 2, 3, 3 }; fill(v.begin(), v.end() ...

  4. 吴裕雄--天生自然HADOOP操作实验学习笔记:hive DDL

    实验目的 了解hive DDL的基本格式 了解hive和hdfs的关系 学习hive在hdfs中的保存方式 学习一些典型常用的hiveDDL 实验原理 有关hive的安装和原理我们已经了解,这次实验我 ...

  5. spring gzip 静态压缩优化

    HTTP 压缩可以大大提高浏览网站的速度,它的原理是,在客户端请求网页后,从服务器端将网页文件压缩,再下载到客户端,由客户端的浏览器负责解压缩并浏览.相对 于普通的浏览过程HTML ,CSS,Java ...

  6. webservice之Http传输错误问题

    1.背景:调用第三方webservice服务,正常调用,但是最近由于第三方更换远程调用地址,并且发布服务器(A)是通过代理的方式请求真实服务器地址(B),于是本以为很简单的将客户端调用地址修改为发布地 ...

  7. Java连载74-字符串常用方法、正则表达式简介

    一.字符串相关的常用方法简介​ package com.bjpowernode.java_learning; ​ public class D74_1_StringCommonMethod { pub ...

  8. perfdog使用手册

    PerfDog(性能狗)测试须知 Android平台 ScreenShot(只支持USB模式) FPS(1秒内游戏画面或者应用界面真实平均刷新次数,俗称帖率/FPS) 1)Avg(FPS):平均帖率 ...

  9. 【你不知道的javaScript 中卷 笔记1】javaScript中的类型与值

    一.类型与值 1.0 javaScript 七种内置类型: 空值(null) 未定义(undefined) 布尔值( boolean) 数字(number) 字符串(string) 对象(object ...

  10. List<SelectListItem> 转为 SelectList

    List<SelectListItem> 转为/转换 SelectList SelectList slPayCh = new SelectList(GetPayChannels(),&qu ...