题目描述

在3×3的棋盘上,摆有八个棋子,每个棋子上标有1至8的某一数字。棋盘中留有一个空格,空格用0来表示。空格周围的棋子可以移到空格中。

要求解的问题是:给出一种初始布局(初始状态)和目标布局(为了使题目简单,设目标状态为123804765)。

找到一种最少步骤的移动方法,实现从初始布局到目标布局的转变。

输入输出格式

输入格式:

输入初始状态,一行九个数字,空格用0表示

输出格式:

只有一行,该行只有一个数字,表示从初始状态到目标状态需要的最少移动次数(测试数据中无特殊无法到达目标状态数据)

输入输出样例

输入:

283104765

输出:

4

分析:

八数码难题,对于我这个C++蒟蒻来说还真有点难,在经过dalao的指点后终于AC了这道题.

先将思路,题面很短相信大家都能看懂,虽然我的代码较长但是应该是最好理解的吧,由题目得知我们有9个数,

其中有一个数字0剩下的数字都能和0交换位置即在数字0能和它的上下左右交换位置(在不超出边界的情况下),

然后我们需要知道当一个矩阵最后交换到"123804765"这种情况的最小步数,最小又是矩阵从这一点我们可以判断出,

此题很有可能用BFS来写,那么问题又来了我们怎么标记?转换成矩阵标记?不不不那样就太麻烦了而且代码也不好实现,

再看题"123804765"其实这已经提示我们可以用字符串来标记,那么怎么标记,我们可以用map<string,bool> vis来实现这个标记功能,

用map就能存下字符型的数组了所以我们每次仅需要判断vis[XX]是否走过就可以完成标记了.

接下来将搜索的实现.

标记我们知道怎么实现,那么关于这道题的搜索又怎么实现?

我们可以由题目得知每个数字只能和0进行交换,我们就可以想到我们可以用0进行对矩阵中每个数字进行更新我们可以用swap实现,

对两个数字进行交换,因为能进行数字交换只能是单个字符的操作所以我们就又得将这些字符全部压进一个新的字符串里这里使用

字符串.push_back(字符)即可实现,然后将新的字符串进行判断是否和"123804765"相同如果相同我们就输出当前的步数,由BFS的最优性

如果满足当前的情况的那就是最优解,注意这里一定要想清楚因为我们每次交换一次每次都更新整个字符串所以当字符串重复时这个情况是不满足的

所我们还要将字符swap回去,当然因为每次交换都会使Map的值发生改变,所以我们也必须swap回去,知道了这些这道题写起来就比较容易了.

在搜索开始前我们当然要进行对这个字符串的预处理当然这个要实现很简单我在这里就不不解释了

更详细的理解请看代码

代码:

#include <iostream>
#include <cstring>
#include <string>
#include <queue>
#include <map>
using namespace std;
int startx,starty,posx=,posy=;//我这个变量名这么清楚可能没人不懂对吧
string s;
char Map[][];
int dirx[]={,,,-};//因为只有四个方向
int diry[]={,,-,};
map<string,bool> vis;//这里我们用map来进行判断重复
struct Node //结构体来存数
{
int x,y,t;
string s;//这里我们声明s来进行判断和更新
};
void bfs()
{
vis[s]=true;//起点肯定要赋值为true啊
queue<struct Node> que;
struct Node now;
now.x=startx;now.y=starty;now.t=;now.s=s;
que.push(now);
while(!que.empty())
{
now=que.front();
que.pop();
posx=,posy=;
for(int i=;i<s.length();i++)//其实这里写在下面比较好理解但不影响
{
Map[posx][posy]=now.s[i];//这里的操作就是我们将我们每次更新完的字符串再用MAP重新赋值再就进行搜索
posy++;
if(posy==)//如果这一排满了
{
posy=;//换下一排
posx++;
}
}
if(now.s=="")//如果满足情况
{
cout<<now.t<<endl;//输出因为BFS最优的特点我们就可以直接输出
return;
}
for(int i=;i<;i++)
{
int xx=dirx[i]+now.x;
int yy=diry[i]+now.y;
if(xx<||xx>||yy<||yy>) continue;//边界处理
swap(Map[now.x][now.y],Map[xx][yy]);
string change;//这里我们需要重新声明一个字符串来进行判断和更新操作
for(int i=;i<=;i++)//因为3*3的矩阵
{
for(int j=;j<=;j++)
change.push_back(Map[i][j]);//将字符全部压进字符串里
}
if(vis[change])//这里需要注意这里很重要当它为重复的情况我们还要swap回去才能continue
{//因为如果我们当前这个不满足情况的值会改变会对后面的搜索产生干扰所以必须swap回去才能continue
swap(Map[xx][yy],Map[now.x][now.y]);
continue;
}
vis[change]=true;
swap(Map[xx][yy],Map[now.x][now.y]);//这里和上面那个是差不多的意思这里因为我们这个BFS和以往的不同因为我们直接是将答案进行了跟新,所以如果我们不更新为上一个状态那会对下一个循环产生影响这个地方难理解需要好好想一下
struct Node next;
next.x=xx;next.y=yy;next.s=change;next.t=now.t+;//这里BFS的基本操作
que.push(next);
}
}
return;
}
int main()
{
cin>>s;
for(int i=;i<s.length();i++)//预处理
{
Map[posx][posy]=s[i];
if(Map[posx][posy]=='')//0为起点
{
startx=posx;//找到它的坐标
starty=posy;
}
posy++;
if(posy==)//当这排y为4时这排就满了因为我是从1开始的所以要到4
{
posy=;//posy赋值为1
posx++;//进入到下一排
}
}
bfs();
return ;
}

洛谷P1379八数码难题的更多相关文章

  1. 洛谷 P1379 八数码难题 解题报告

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

  2. 洛谷——P1379 八数码难题

    P1379 八数码难题 双向BFS 原来双向BFS是这样的:终止状态与起始状态同时入队,进行搜索,只不过状态标记不一样而已,本题状态使用map来存储 #include<iostream> ...

  3. 洛谷 P1379 八数码难题 Label:判重&&bfs

    特别声明:紫书上抄来的代码,详见P198 题目描述 在3×3的棋盘上,摆有八个棋子,每个棋子上标有1至8的某一数字.棋盘中留有一个空格,空格用0来表示.空格周围的棋子可以移到空格中.要求解的问题是:给 ...

  4. 洛谷 P1379 八数码难题

    题目描述 在3×3的棋盘上,摆有八个棋子,每个棋子上标有1至8的某一数字.棋盘中留有一个空格,空格用0来表示.空格周围的棋子可以移到空格中.要求解的问题是:给出一种初始布局(初始状态)和目标布局(为了 ...

  5. 洛谷 - P1379 - 八数码难题 - bfs

    https://www.luogu.org/problemnew/show/P1379 #include <bits/stdc++.h> using namespace std; #def ...

  6. 洛谷—— P1379 八数码难题

    https://daniu.luogu.org/problem/show?pid=1379 题目描述 在3×3的棋盘上,摆有八个棋子,每个棋子上标有1至8的某一数字.棋盘中留有一个空格,空格用0来表示 ...

  7. 洛谷P1379 八数码难题

    传送门 1.先用dfs枚举9!的全排列,存到hash数组里(类似离散化),因为顺序枚举,就不需要排序了 2.朴素bfs,判重就用二分找hash:如果发现当前状态=要求状态,输出步数结束程序 上代码 # ...

  8. 洛谷 P1379 八数码难题 题解

    我个人感觉就是一道bfs的变形,还是对bfs掌握不好的人有一定难度. 本题思路: 大体上用bfs搜,用map来去重,在这里只需要一个队列,因为需要较少步数达到的状态一定在步数较多的状态之前入队列. # ...

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

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

随机推荐

  1. 【luogu P2939 [USACO09FEB]改造路Revamping Trails】 题解

    题目链接:https://www.luogu.org/problemnew/show/P2939 本来说是双倍经验题,跟飞行路线一样的,结果我飞行路线拿deque优化SPFA过了这里过不了了. 所以多 ...

  2. JVM 监控以及内存分析

    1 内存分析1.1 jmap -histo 命令pid=`jps | awk '{if ($2 == "Jps") print $1}'`jmap -histo $pid > ...

  3. $_GET 与 $POST

    $_GET就是地址传值,用 '?' 开始传值,多个值间用 '&' 号分隔,多用于简单的传值,比如说看新闻需要新闻id一般就会用地址传值, $_GET的好处是传值可见,也就是只要一个地址就ok了 ...

  4. 常用的linux指令整理

    ls 列出文件目录 -a全部文件,连同隐藏文件一起列出 -d仅列出目录本身,而不是列出目录内的文件数据 -l 连同权限一同列出 cd 切换文件目录的命令 pwd显示目前所在的目录 mkdir 创建新的 ...

  5. antd不想写那么多option怎么办

    做项目的时候发现如果下拉列表选项多的时候会写很多的 Option ,但是用到下拉列表的地方又超级多.所以自己写了一个方法,哪需要就放到哪. 记录一下方法.留待以后用 selectStreetIdCha ...

  6. 自动诊断档案库(ADR)学习

    (1)ADR概述 Oracle 11g的FDI(Fault Diagnosability Infrastructure)是自动化诊断方面的一个增强,其核心组件为自动诊断库(Automatic Diag ...

  7. display:flex 布局之 骰子

    代码部分 html <body> <div class="box"> <div class="a a1"> <span ...

  8. D - 湫湫系列故事——减肥记II

    虽然制定了减肥食谱,但是湫湫显然克制不住吃货的本能,根本没有按照食谱行动! 于是,结果显而易见… 但是没有什么能难倒高智商美女湫湫的,她决定另寻对策——吃没关系,咱吃进去再运动运动消耗掉不就好了? 湫 ...

  9. python运算符及优先级顺序

    python语言是一门脚本语言,支持面向对象.面向过程编程,兼具编译性和解释性的动态语言,整理出学习过程中一些基本Python运算符和运算符的优先级顺序. 一.算术运算符 运算符 描述 + 加 - 两 ...

  10. (转)程序员新人怎样在复杂代码中找 bug?

    我曾经做了两年大型软件的维护工作,那个项目有10多年了,大约3000万行以上的代码,参与过开发的有数千人,代码checkout出来有大约5个GB,而且bug特别多,open的有上千,即使最高优先级的s ...