题目描述

如下面第一个图的九宫格中,放着  1~8  的数字卡片,还有一个格子空着。与空格子相邻的格子中的卡片可以移动到空格中。经过若干次移动,可以形成第二个图所示的局面。

我们把第一个图的局面记为:12345678.
把第二个图的局面记为:123.46758
显然是按从上到下,从左到右的顺序记录数字,空格记为句点。
本题目的任务是已知九宫的初态和终态,求最少经过多少步的移动可以到达。如果无论多少步都无法到达,则输出-1。

输入格式

输入第一行包含九宫的初态,第二行包含九宫的终态。

输出格式

输出最少的步数,如果不存在方案,则输出-1。

样例输入

12345678.
123.46758 

样例输出

3

解题方法

BFS来搜索目标局面,一旦搜到一定是最小移动次数。

通常利用哈希表记录每一种不同的局面

typedef int State[9];//哈希映射
State st[Maxsize],goal;//st二维数组存储每种情形,goal数组存储目标数组
int hash(State& s)
{int v = 0;for(int i=0; i<9; i++) v = v*10 + s[i]; //把9个数字组合成9位数 return     v % Maxsize; //确保hash函数值是不超过hash表的大小的非负整数
}

复习一下三个函数

memcpy(&t ,&s ,sizeof(s));//从源source所指的内存地址的起始位置开始拷贝n个字节到目标destin所指的内存地址的起始位置中。
if(memcmp(st[s], st[u], sizeof(st[s]))==0) return 0; //把存储区str1和存储区str2的前n个字节进行比较
memset(head, 0, sizeof(head));//为新申请的内存做初始化工作。

插入链表的实现

int try_to_insert(int s)
{int h = hash(st[s]); int u = head[h]; //从表头开始查找链表 while(u){//把存储区str1和存储区str2的前n个字节进行比较if(memcmp(st[s], st[u], sizeof(st[s]))==0) return 0; //有重复,插入失败 u = next[u]; //顺着链表继续找 }next[s] = head[h]; //该结点插入到链表中 head[h] = s;return 1;
}

核心bfs实现

int bfs()
{init_table();int front=1,rear=2;while(front<rear){State &s=st[front];int z;if(memcmp(goal,s,sizeof(s))==0) return front;for(int i=0;i<9;i++){if(!s[i]) z=i;}int x=z/3,y=z%3;for(int i=0;i<4;i++){int newx=x+dx[i];int newy=y+dy[i];int newz=3*newx+newy;if(newx<3&&newx>=0&&newy<3&&newy>=0){State &t=st[rear];memcpy(&t,&s,sizeof(s));t[newz]=s[z];t[z]=s[newz];dist[rear]=dist[front]+1;if(try_to_insert(rear)) rear++;}}front++;}return 0;
}

参考代码

#include<iostream>
#include<cstring>
using namespace std;
const int Maxsize=1000000;
string str1,str2;
int dist[Maxsize];
int dx[4]={0,0,1,-1};
int dy[4]={1,-1,0,0};
typedef int State[9];
State st[Maxsize],goal;
int head[Maxsize],next1[Maxsize];
void init_table()
{memset(head,0,sizeof(head));
}
int hash1(State &s)
{int sum=0;for(int i=0;i<9;i++){sum=sum*10+s[i];}return sum%Maxsize;
}
int try_to_insert(int s)
{int h=hash1(st[s]);int u=head[h];while(u){if(memcmp(st[s],st[u],sizeof(st[s]))==0)return 0;u=next1[u];}next1[s]=head[h];head[h]=s;return 1;
}
int bfs()
{init_table();int front=1,rear=2;while(front<rear){State &s=st[front];int z;if(memcmp(goal,s,sizeof(s))==0) return front;for(int i=0;i<9;i++){if(!s[i]) z=i;}int x=z/3,y=z%3;for(int i=0;i<4;i++){int newx=x+dx[i];int newy=y+dy[i];int newz=3*newx+newy;if(newx<3&&newx>=0&&newy<3&&newy>=0){State &t=st[rear];memcpy(&t,&s,sizeof(s));t[newz]=s[z];t[z]=s[newz];dist[rear]=dist[front]+1;if(try_to_insert(rear)) rear++;}}front++;}return 0;
}
int main()
{cin>>str1>>str2;for(int i=0; i<9; i++) {if(str1[i]=='.') st[1][i] = 0;elsest[1][i] = str1[i] - '0';}for(int i=0; i<9; i++){if(str2[i]=='.')goal[i] = 0;elsegoal[i] = str2[i] - '0';} int ans = bfs();if(ans == 0) printf("%d\n",-1);else printf("%d\n",dist[ans]);return 0;
}

蓝桥杯历届试题-九宫重排相关推荐

  1. 历届试题+九宫重排+java_蓝桥杯 历届试题 九宫重排 (bfs+康托展开去重优化)...

    Description 如下面第一个图的九宫格中,放着 1~8 的数字卡片,还有一个格子空着.与空格子相邻的格子中的卡片可以移动到空格中.经过若干次移动,可以形成第二个图所示的局面. 我们把第一个图的 ...

  2. 蓝桥杯 历届试题 九宫重排 (bfs+康托展开去重优化)

    Description 如下面第一个图的九宫格中,放着 1~8 的数字卡片,还有一个格子空着.与空格子相邻的格子中的卡片可以移动到空格中.经过若干次移动,可以形成第二个图所示的局面. 我们把第一个图的 ...

  3. java九宫排序,蓝桥杯 历届试题 九宫重排 java

    list = new ArrayList(); list.add(new Move(x1,y1,0,start));//添加初始状态进入list set.add(start);//添加初始字符串进入s ...

  4. 蓝桥杯 历届试题 九宫重排

    问题描述 如下面第一个图的九宫格中,放着 1~8 的数字卡片,还有一个格子空着.与空格子相邻的格子中的卡片可以移动到空格中.经过若干次移动,可以形成第二个图所示的局面. 我们把第一个图的局面记为:12 ...

  5. [蓝桥杯][历届试题]九宫重排-双向bfs和map标记

    题目描述 如下面第一个图的九宫格中,放着 1~8 的数字卡片,还有一个格子空着.与空格子相邻的格子中的卡片可以移动到空格中.经过若干次移动,可以形成第二个图所示的局面. 我们把第一个图的局面记为:12 ...

  6. 蓝桥杯 历届试题 九宫重排

    问题描述 如下面第一个图的九宫格中,放着 1~8 的数字卡片,还有一个格子空着.与空格子相邻的格子中的卡片可以移动到空格中.经过若干次移动,可以形成第二个图所示的局面. 我们把第一个图的局面记为:12 ...

  7. python 九宫重排_[蓝桥杯][历届试题]九宫重排 (Python代码)(bfs+集合)

    ac代码..可是官网还有两组没过.. ```python from collections import deque class Node(): def __init__(self, arr, ste ...

  8. 蓝桥杯 历届试题 九宫幻方

    问题描述   小明最近在教邻居家的小朋友小学奥数,而最近正好讲述到了三阶幻方这个部分,三阶幻方指的是将1~9不重复的填入一个3*3的矩阵当中,使得每一行.每一列和每一条对角线的和都是相同的. 三阶幻方 ...

  9. 算法笔记_183:历届试题 九宫重排(Java)

    目录 1 问题描述 2 解决方案   1 问题描述 问题描述 如下面第一个图的九宫格中,放着 1~8 的数字卡片,还有一个格子空着.与空格子相邻的格子中的卡片可以移动到空格中.经过若干次移动,可以形成 ...

最新文章

  1. 终于有人解救Python人了!
  2. 去掉一个linux的ip,linux – iptables删除除一个IP之外的所有传入ICMP请求
  3. 学习jquery的一些笔记
  4. OpenSSL“心脏出血”漏洞
  5. linux grpc,grpc linux下的编译使用-Go语言中文社区
  6. telnet到设备里 php_金融行业思科设备典型网络故障案例:76系列典型案例(一)...
  7. 疯狂VirtualBox实战讲学录 以及 virtualbox完全学习手册 之我见
  8. fragment photoshop_史上最接地气的Photoshop?谈PS 2021的黑科技 - Adobe
  9. 《Android移动应用开发》 复习题(一)
  10. fanuc服务器显示8,FANUC伺服驱动器报故障数字8的维修一看就懂
  11. 推荐一部非常精彩的小说
  12. RecyclerView条目item点击事件---放大
  13. Linux 内存性能检测工具
  14. Java中的逻辑控制与方法(上)
  15. google chrome 谷歌浏览器 清除 清空 缓存cookie 快捷键
  16. 计算机专业网名英语翻译,英文网名_好听的英文网名带翻译
  17. python 股票_十分钟学会用Python交易股票
  18. Windows XP最后的安全补丁
  19. eval python定义变量,表达式与语句 局部变量与全局变量 exec与eval
  20. Flutter开发学习课程携程app开发(完)

热门文章

  1. 最近的热门:渣打小三事件
  2. 非常值得看的一篇文章—HP大中华区总裁孙振耀退休感言
  3. WPF下聊天气泡的实现
  4. 【踩坑记录】—— 越南语ipa包安装失败
  5. qq视频转码失败怎么办_腾讯视频怎么转码
  6. 趁年轻要学会打破思维
  7. 一文回顾AI绘画的成长之路:从简笔画到真实人脸生成
  8. 实习时候的亚子==(四)
  9. 盘点好莱坞电影对人工智能的那些错误理解!
  10. 小小白学Springboot(十一)——Security安全管理