问题描述

九宫格拼图就是在3×3的格子上摆放8块拼图,空出1个格子,玩家要借助这1个空格上下左右滑动拼图,最终完成整幅图画

我们像下面这样将空格定为0,然后给8块拼图分别标上1到8号

1 3 0
4 2 5
7 8 6

1次操作可以将1块拼图移向空格,当8块拼图全部与下述位置吻合时完成游戏

1 2 3
4 5 6
7 8 0

现给定九宫格拼图的初始状态,请编写一个程序,求出完成该九宫格拼图最少需要移动多少次

输入:
输入表示拼图块或空格的3×3个整数。每3个整数占1行,总计3行,相邻数据间用空格隔开
输出:
输出最少移动次数,占1行
限制:
给定拼图必然有解。

输入示例

1 3 0
4 2 5
7 8 6

输出示例

4

讲解

搜索算法可以通过重复进行“状态迁移”来寻求拼图的解法,也就是说,我们可以利用搜索算法求解这类拼图问题。一般来说,搜索算法会生成一个从初始状态到最终状态(完成状态)的状态变化序列。而在九宫格拼图问题中,我们还需要从所有可能的序列中选择最短的一条

为避免重复生成相同的状态,大部分搜索算法的搜索空间都为树结构,树的结点表示状态,边表示状态迁移

九宫格拼图中,各拼图块和空格的位置就是“状态”,上下左右移动拼图块相当于“状态迁移”。要想有效管理各个状态(拼图格局)的生成情况,需要应用散列法或二叉搜索树

像九宫格拼图这种状态总数不多的问题,我们完全可以用深度优先搜索或广度优先搜索来求解

这里的深度优先搜索与图的深度优先搜索原理相同(参见:深度优先搜索 | DFS | Depth First Search | C/C++实现),从初始状态出发,尽可能地进行状态迁移,直至抵达最终状态为止。不过,一旦搜索遇到类似于下面这样的情况,就中断当前搜索并返回上一状态:

当前状态无法再进行状态迁移时
状态迁移生成了曾经生成过的状态时
根据问题的性质可断定继续搜索无法找到答案时

说白了就是进行回溯。这种打断搜索的处理还可以形象地称为剪枝
有限制的深度优先搜索称为深度受限搜索。该算法会在搜索深度(树的深度)达到某个既定值limit时中断搜索。也就是说,如果能根据问题的性质限制搜索深度,我们就能进一步提高搜索的效率

单纯的深度优先搜索算法具有下列特征:
求出来的不一定是最短解
可能因为部分无用的搜索导致复杂度上升
如果不进行剪枝,在最坏情况下(无解等情况)会变成穷举搜索


这里的广度优先搜索与图的广度优先搜索原理相同(参见:广度优先搜索 | BFS | Breadth First Search | C/C++实现),算法会尽可能广地进行状态迁移

从当前状态出发,进行所有有可能的状态迁移并得到新状态。为能够系统地进行搜索,我们需要将状态迁移后生成的新状态添加至队列,同时从队列开头的状态进一步进行状态迁移,使得搜索不断展开。为防止生成已有状态,我们还需要将已生成状态记录在内存中

广度优先搜索需要占用大量内存,但只要问题有解,就可以相对简单地搜索到初始状态最终状态的最短路径

AC代码如下

#include<iostream>
#include<cmath>
#include<string>
#include<map>
#include<queue>
using namespace std;
#define N 3
#define N2 9struct Puzzle {int f[N2];int space;string path;bool operator < (const Puzzle &p) const {for(int i = 0; i < N2; i++) {if(f[i] == p.f[i]) continue;return f[i] > p.f[i];}return false;}
};static const int dx[4] = {-1, 0, 1, 0};
static const int dy[4] = {0, -1, 0, 1};
static const char dir[4] = {'u', 'l', 'd', 'r'};bool isTarget(Puzzle p) {for(int i = 0; i < N2; i++)if(p.f[i] != (i + 1) ) return false;return true;
}string bfs(Puzzle s) {queue<Puzzle> Q;map<Puzzle, bool> V;Puzzle u, v;s.path = "";Q.push(s);V[s] = true;while(!Q.empty() ) {u = Q.front(); Q.pop();if(isTarget(u) ) return u.path;int sx = u.space / N;int sy = u.space % N;for(int r = 0; r < 4; r++) {int tx = sx + dx[r];int ty = sy + dy[r];if(tx < 0 || ty < 0 || tx >= N || ty >= N) continue;v = u;swap(v.f[u.space], v.f[tx * N + ty]);v.space = tx * N + ty;if(!V[v]) {V[v] = true;v.path += dir[r];Q.push(v);}}}return "unsolvable";
}int main(){Puzzle in;for(int i = 0; i < N2; i++){cin>>in.f[i];if(in.f[i] == 0) {in.f[i] = N2; //set spacein.space = i;}}string ans = bfs(in);cout<<ans.size()<<endl;return 0;
}

九宫格拼图 | 8Puzzle | C/C++实现相关推荐

  1. 算法可视化器 | 九宫格拼图 | 8Puzzle

    文章目录 项目环境 应用功能 对于处理八块拼图问题的六种算法描述 DFS BFS UCS GBFS A* IDA* 部分截图 状态输入 DFS搜索树 GBFS搜索树 參考 项目环境 编译器: Netb ...

  2. android实现九宫格拼图小游戏

    贴一下效果图 接下来随便用一张图片就好 以下是全代码 自定义View GameView类 import android.content.Context; import android.graphics ...

  3. 九宫格拼图怎么拼?分享两个简单的操作

    日常生活中发朋友圈的时候,小伙伴是不是有很多照片想分享到朋友圈呢?但是因为朋友圈的限制,不能一次全发.有很多朋友发现很多朋友可以把一张图片剪成9格!看起来很棒,那九宫格拼图怎么拼的呢?今天就和大家分享 ...

  4. java设计九宫格拼图软件哪个好用_九宫格拼图软件下载_抖音很火的九宫格拼图软件app下载_易玩网...

    最近抖音里面好多小伙伴都在晒自己的九宫格拼图照片,小编就为您寻找到了这款APP,软件名字就是分图,能够轻松实现九宫格拼图的制作,你可以随时选择九宫格或者二宫格.三宫格.四宫格以及六宫格,轻松一键制作你 ...

  5. 软件工程作业---结对编程の九宫格拼图扩展

    程序下载:九宫格拼图扩展.exe 想必大家都熟悉九宫格拼图,传统的九宫格拼图就是九个格子,而其中有一个是空的,旁边的可以移动到空的位置,然后通过移动来把乱的图片拼成一个完整的图片,看似简单但其中又有技 ...

  6. java编写九宫格拼图游戏_九宫格拼图游戏

    九宫格拼图游戏设计文档 一.综合设计目的.条件.任务和内容要求: 1.设计目的 <Windows程序设计>是计算机科学与技术专业本科生的一门学科基础课程.Windows程序以图形用户界面( ...

  7. 九宫格拼图游戏设计,及代码时序问题解决

    目录 1. 需求简述 2. 大方向思路 3. 具体实现思路 4. 问题描述 5. 问题解决 1. 需求简述 需求是一个九宫格拼图的游戏,每两张图都可以随意对换,当拖动图片经过被交换图片时,拖动图片不动 ...

  8. VB实现可调节难度的九宫格拼图

    #VB实现可调节难度的九宫格拼图 实现本游戏需要熟练掌握paintpicture函数的应用,如果读者对该函数不熟练,需要读一下前面两篇博客哦 http://blog.csdn.net/wf824284 ...

  9. java设计九宫格拼图软件哪个好用_十亿人都在拼的拼图软件,这八款最好用

    Hey,各位喵友好,我是33. 作为一个喜欢拍照的人,不仅要熟悉掌控各种美颜滤镜APP,保证自己能随时P好照片发朋友圈,更要让自己的朋友圈也精致起来~ 可是,简单地发图片功效其实是不够滴,(而且万一想 ...

  10. 九宫格拼图游戏初版(练练手)

    最近自己看完了java初级篇,课堂上讲的太少了,有那么一会儿灵感蹦出就写了九宫格游戏,只是初版,有些细节未处理,但整体功能都已实现,贴截图: 程序有三个.class文件,依次是程序主体Nine,图片方 ...

最新文章

  1. 0x63.图论 - 树的直径与最近公共祖先
  2. 服务器高并发时请求报错_基于redis的分布式锁防止高并发重复请求
  3. stm32 窗口看门狗学习(一)
  4. PyTorch基础-Adam优化器使用-06
  5. 业界萌新对斯坦纳树的小结
  6. 如何在一周内上线50个用户增长策略
  7. jmeter 获取全部响应,jmeter中的正则表达式提取器-从响应中提取多个值.
  8. HandyJSON:Swift语言JSON转Model工具库
  9. 八.nginx网站服务实践应用
  10. python搜索word关键字_Python根据关键字抓取word相关内容
  11. 正则表达式判断手机号码
  12. access如何保存小数点后_你知道PDF文件旋转页面后如何保存吗?
  13. 第七章 变量进阶与点阵LED 练习题
  14. ThinkPad E420升级之路
  15. TextView添加中划线、下划线等
  16. 【网络工程】浅显易懂TCP/IP协议 三次握手 四次挥手
  17. 石墨计算机,石墨文档电脑版
  18. 阿里云SLB最佳实践
  19. 怎样把ogg格式转换mp3
  20. 【JavaSE】集合

热门文章

  1. uniapp引入腾讯防水墙
  2. 【历史上的今天】10 月 17 日:微软发布 Windows 8.1;IMDb 成立;海盗湾创始人诞生
  3. ImportError: cannot import name ‘PILLOW_VERSION‘ from ‘PIL‘ (/home/user8/anaconda3/envs/FCOS/lib/pyt
  4. python问题 Traceback (most recent call last)
  5. 计算机无法访问家庭组内打印机,Win7电脑无法连接共享打印机拒绝访问怎么办...
  6. 超宽屏幕比例_Hello!宽时代 21:9超宽屏显示器选购
  7. 英语魔法师之语法俱乐部阅读笔记
  8. 转自登峰之群:晓军教材(一)
  9. 优化计算机组策略,windows系统优化--使你的计算机飞起来
  10. Linux 别名设置,可一键登入服务器- alias