九宫问题(八数码)求解过程动态演示

无聊在茅坑逗蛆

无聊在茅坑逗蛆

2016-01-29 12:15

九宫问题(八数码)求解过程动态演示,九宫问题(八数码)求解过程动态演示

九宫问题(八数码)求解过程动态演示

作者:赵宏伟

(本文来源于图老师网站,更多请访问http://m.tulaoshi.com/cyuyanjiaocheng/) 下载源代码

一、题目说明:
  
(九宫问题)在一个3×3的九宫中有1-8这8个数及一个空格随机的摆放在其中的格子里,如图1-1所示。现在要求实现这个问题:将该九宫格调整为如图1-1右图所示的形式。调整的规则是:每次只能将与空格(上、下、或左、右)相邻的一个数字平移到空格中。试编程实现这一问题的求解。

(图1-1)

二、题目分析:

  
九宫问题是人工智能中的经典难题之一,问题是在3×3方格棋盘中,放8格数,剩下的没有放到的为空,每次移动只能是和相邻的空格交换数。程序自动产生问题的初始状态,通过一系列交换动作将其转换成目标排列(如下图1-2到图1-3的转换)。
  (图1-2) (图1-3)

  九宫问题中,程序产生的随机排列转换成目标共有两种可能,而且这两种不可能同时成立,也就是奇数排列和偶数排列。我们可以把一个随机排列的数组从左到右从上到下用一个一维数组表示,如上图1-2我们就可以表示成{8,7,1,5,2,6,3,4,0}其中0代表空格。
在这个数组中我们首先计算它能够重排列出来的结果,公式就是:

∑(F(X))=Y,其中F(X)

  就是一个数他前面比这个数小的数的个数,Y为奇数和偶数个有一种解法。那么上面的数组我们就可以解出它的结果。

F(8)=0;F(7)=0;F(1)=0;F(5)=1;F(2)=1;F(6)=3;F(3)=2;F(4)=3;Y=0+0+0+1+1+3+2+3=10

  Y=10是偶数,所以他的重排列就是如图1-3的结果,如果加起来的结果是奇数重排的结果就是如图1-1最右边的排法。

三、算法分析

  
九宫问题的求解方法就是交换空格(0)位置,直至到达目标位置为止。图形表示就是:

(本文来源于图老师网站,更多请访问http://m.tulaoshi.com/cyuyanjiaocheng/)

(图3-1)

  要想得到最优的就需要使用广度优先搜索,九宫的所以排列有9!种,也就是362880种排法,数据量是非常大的,我使用的广度搜索,需要记住每一个结点的排列形式,要是用数组记录的话会占用很多的内存,我们把数据进行适当的压缩。使用DWORD形式保存,压缩形式是每个数字用3位表示,这样就是3×9=27个字节,由于8的二进制表示形式1000,不能用3位表示,我使用了一个小技巧就是将8表示位000,然后用多出来的5个字表示8所在的位置,就可以用DWORD表示了。用移位和或操作将数据逐个移入,比乘法速度要快点。定义了几个结果来存储遍历到了结果和搜索完成后保存最优路径。
类结构如下:

class CNineGird{public:struct PlaceList    {DWORDPlace;PlaceList*Left;PlaceList*Right;    };struct Scanbuf{DWORD Place;int ScanID;};struct PathList{unsigned char Path[9];};private:PlaceList *m_pPlaceList;Scanbuf *m_pScanbuf;RECT m_rResetButton;RECT m_rAutoButton;public:int m_iPathsize;clock_t m_iTime;UINT m_iStepCount;unsigned char m_iTargetChess[9];unsigned char m_iChess[9];HWND m_hClientWin;PathList *m_pPathList;bool m_bAutoRun;private:inline bool AddTree(DWORD place , PlaceList*& parent);void FreeTree(PlaceList*& parent);inline void ArrayToDword(unsigned char *array , DWORD & data);inline void DwordToArray(DWORD data , unsigned char *array);inline bool MoveChess(unsigned char *array , int way);bool EstimateUncoil(unsigned char *array);void GetPath(UINT depth);public:void MoveChess(int way);bool ComputeFeel();void ActiveShaw(HWND hView);void DrawGird(HDC hDC , RECT clientrect);      
展开更多 50%)
分享

猜你喜欢

九宫问题(八数码)求解过程动态演示

C语言教程 C语言函数
九宫问题(八数码)求解过程动态演示

八皇后问题求解

编程语言 网络编程
八皇后问题求解

s8lol主宰符文怎么配

英雄联盟 网络游戏
s8lol主宰符文怎么配

《神雕侠侣》装备祝福冲星一元至九宫所需材料

手机游戏 IOS
《神雕侠侣》装备祝福冲星一元至九宫所需材料

模拟退火算法求解TSP问题

C语言教程 C语言函数
模拟退火算法求解TSP问题

lol偷钱流符文搭配推荐

英雄联盟 网络游戏
lol偷钱流符文搭配推荐

mysql 动态执行存储过程语句

编程语言 网络编程
mysql 动态执行存储过程语句

十二星座: 求解塔罗牌问题六芒星牌阵

十二星座 星座
十二星座: 求解塔罗牌问题六芒星牌阵

lolAD刺客新符文搭配推荐

英雄联盟
lolAD刺客新符文搭配推荐

简单二叉树类

简单二叉树类

Flash AS3.0菜鸟学飞教程:类的编写之不使用库元件

Flash AS3.0菜鸟学飞教程:类的编写之不使用库元件
下拉加载更多内容 ↓