关闭

老调重提,利用 SDK 实现迷宫算法

标签: 算法数据结构windowsstructtimersearch
976人阅读 评论(0) 收藏 举报
我近来重看了数据结构的书,现在的教材还是使用C/C++的编写的算法,编译还是在console mode进行, 如果能把这些数据结构的算法使用在SDK上,那么就可以开发出 Windows 程序的算法程序提高学习,不用在 单调的console mode 中看着冷冰冰的字符来学习数据结构了,这样学习一方面可以学习调用 Windows API 和 Windows编程,另一方面可以学习数据结构. 希望我这样的学习方法对那些初学 Windows 的朋友有一些帮助.
这是使用 SDK 开发出来的迷宫程序(F1 键开始).

迷宫算法还是老路子,回溯法和堆栈实现,我采用的是堆栈实现. 使用双向链表摸拟堆栈,使用一个 ptrFirst 和一个 ptrLast 和作为堆栈的栈底和栈顶指针, 定义一 个堆栈元素结构, 这个结构保存迷宫中的位置.
typedef struct _tagNode {
	int	nRow;
	int	nColumn;
	struct _tagNode* next;
	struct _tagNode* previou;
} Node;
定义一个标记数组
BOOL bPass[ Row ][ Column ];	// Row 和 Column 为迷宫大小.
迷宫算法的主要伪代码的实现方式.
A.从开始位置开始,判断小球的各个方向是否可行,若一个方向可行,则向该方向移动.
前进的位置进栈.
条件: 前进方向是墙, 则该方向不能向前.
前进和方向如果是经过的,则该方向不能向前.
	if ( CanMove( gnRow, gnColumn, right ) )
	{
		gnColumn += moveRight; // 前进
		bPass[ gnRow ][ gnColumn ] = TRUE; // 标记通过的位置
		// gnRow, gnColumn 位置入栈.
	}
	else if ( CanMove( gnRow, gnColumn, left ) ) // right 方不通, 向左.
	{
		gnColumn += moveLeft; // 前进
		bPass[ gnRow ][ gnColumn ] = TRUE; // 标记通过的位置
		// gnRow, gnColumn 位置入栈.
	}
	else if ( CanMove( gnRow, gnColumn, forward ) ) // left 方不通, 向前.
	{
		gnColumn += moveForward; // 前进
		bPass[ gnRow ][ gnColumn ] = TRUE; // 标记通过的位置
		// gnRow, gnColumn 位置入栈.
	}
	else if ( CanMove( gnRow, gnColumn, back ) ) // forward 方不通, 向后.
	{
		gnColumn += moveBack; // 前进
		bPass[ gnRow ][ gnColumn ] = TRUE; // 标记通过的位置
		// gnRow, gnColumn 位置入栈.
	}
B.各个方向不能可行,退回前一个位置,利用退栈操作,回到 A.
	else if ( CanMove( gnRow, gnColumn, back ) )
	{}
A, B 不断重复, 直到找到出口, 或遍历迷宫(栈空)
	if 为迷宫出口
		bSearch = FALSE;
	if 栈为空 // 没有出口,
		bSearch = FALSE;
从这个算法我们就可以利用 SDK 来实现这个迷宫, 但是有几个问题必须要注意的,第一个, 在纯 C/C++开发中(不调用API) 是,我们的循环是使用 while( 1 ) { ...... } 来实现的,但是在 Windows 编程之 中,每个 Windows 程序是消息驱动的( Event driven ), 它本身就是无限循环的,这样一来, 你要改变一 下你的想法, 我们不使用 while( 1 ){ .... } 来实现循环, 要用消息来实现循环, 这个消息是 Windows 程序自已发出的, 我们不用添加自定义的消息. ::SetTimer( .... ); 可以每隔固定时间发出一个 WM_TIMER 的消息, 这样我们就可以利用这个消息来实现循环, 因为这每隔固定时间就有一个消息, 所以我们可以利用这个消息控制小球的速度. 利用一个 flag 来判断循环是否可以结束, 而不使用 break.
case WM_TIMER:
	if ( bStart )
		Start();
	if ( bSearch )
		Search();	// bSearch 循环结束标志. 找到出口或栈为空, bSearch = FALSE.
	return 0; 

其次第二个注意的问题, 在 SDK 中, 使用的数据都是 static 或 global 的, 所以对全局数据的操作地方(读写操作)最好能够在一个函数内完成, 避免过多的使用修改函数而使变量条理不清, 在源代码中你可以看 到, 在 Search() 中, 只对全局变量 ptrFirst, ptrLast, gnRow, gnColumn 操作, 其它函数不负责数据的 操作.

第三个问题, 资源分配释放的问题, 对于占用的资源, 在接到 WM_DESTROY 消息进行内存释放.

剩下的,就只需要学习贴图的技巧就行了, 这些比较简单, 只要看看源代码就可以明白的了, 我只是在这儿 说明怎样造成小球的运动效果, 小球一旦移动时, 把它的移动前位置屏蔽掉, 就这样地重复过程就可以造成 运动效果.

附带的源代码中有两个贴图的小程序,写得尽量简单,对刚学习 Windows 编程的朋友一定有帮助. 编译时只要在 console mode 输入 nmake 命令即可进行编译, 把所学用到所用,这样的学习数据结构就是一个很有乐趣的过程.

源代码http://www.vckbase.com/code/downcode.asp?id=2448

0
0
查看评论

老调重提,利用SDK实现智能五子棋

网上有很多的实现五子棋的算法,如利用规则法,递归法,博弈树法来实现五子棋的,上次我写了一篇利用SDK实现迷宫算法的文章,这次还是同样的题目,老调重提,我利用的是规则法来实现五子棋的智能。不过我个人认为还是博弈树法还是简洁。如果读者对博弈树有兴趣的,可以重读数据结构中的树结构的实现这一部分! 这是利用...
  • laiboy
  • laiboy
  • 2004-09-27 00:21
  • 1204

利用堆栈实现迷宫问题的求解

的说法是
  • meiyubaihe
  • meiyubaihe
  • 2014-06-07 10:47
  • 1338

迷宫求解算法(栈DFS以及队列BFS)

我们首先给出一个迷宫,它的规格是5 * 5,在这里我使用int的二维数组表示迷宫,其中1表示障碍,0表示可以通行的道路,要求从(0,0)坐标走到(4, 4)坐标,并输出走过的坐标路径。int maze[5][5] = { 0, 1, 0, 0, 0, 0, 1, 0, 1, 0, ...
  • laughing2333
  • laughing2333
  • 2015-11-03 17:35
  • 1614

算法:堆栈与深度优先搜索(迷宫问题)

堆栈的访问规则被限制为Push和Pop两种操作,Push(入栈或压栈)向栈顶添加元素,Pop(出栈或弹出)则取出当前栈顶的元素,也就是说,只能访问栈顶元素而不能访问栈中其它元素。 现在我们用堆栈解决一个有意思的问题,定义一个二维数组: int maze[5][5] = {  &#...
  • Simba888888
  • Simba888888
  • 2013-04-27 16:26
  • 7883

DFS算法之迷宫寻路问题

要求输入两个整数m,n表示迷宫矩阵大小(m*n),然后输入迷宫矩阵,0表示死路,1表示通路。令迷宫入口坐标为(0,0)出口坐标为(m-1,n-1)。 要求输出走出迷宫的所有路线和最短的一条路线。 如: 输入 4 4 1 1 1 1 0 1 1 0 1 1 1 0 0 1 1 1 则输出 找到路线:...
  • sdzhr
  • sdzhr
  • 2017-03-19 20:49
  • 907

迷宫算法(c语言实现)

利用c语言实现迷宫算法,环境是vc++6.0.#include #include#include int visit(int,int); void setmaze();int maze[11][11]={    {0,0,2,2,2,2,2,2,2,2}, &...
  • reghi
  • reghi
  • 2010-04-29 20:25
  • 2367

迷宫最短路径算法(使用队列)

    (上接迷宫递归算法)    顺便也把图里求迷宫最短路径算法贴出来,主要思想是利用队列,采用广度优先搜索法,当第一次出现目的点时,中断搜索,并输出路径。程序还是主要使用的C语言,对于队列操作我又重写了下基本操作代码比如入队、出队等,没...
  • littlehedgehog
  • littlehedgehog
  • 2007-08-25 13:17
  • 8415

C语言广度优先搜索之迷宫(队列)

#include #define MAX_ROW 5 #define MAX_COL 5 struct point { int row, col, predecessor; } queue[512]; int head = 0, tail = 0; void enqueue(struct poin...
  • e421083458
  • e421083458
  • 2014-09-12 13:02
  • 4420

用栈实现简单的迷宫游戏

迷宫游戏的实现是运用了栈的“后进先出”的原理,究竟如何实现的呢?下来简单的分析一下。 比如:下图是一个小迷宫 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 0 0 0 1 1 1 1 1 1 1 1 1 0 1 1 1 1 1 1 1 //0---...
  • qq_29503203
  • qq_29503203
  • 2016-09-10 16:07
  • 479

用栈和递归求解迷宫问题

一、问题概述 小时候,我们都玩过走迷宫的游戏吧。看一下这个图例: 遇到这种问题时,我们第一反应都会先找到迷宫的入口点,然后对上下左右四个方向进行寻迹,  检测当前位置是否是通路,是否可以通过,直至找到出口位置,才是迷宫的正确轨迹。如若走到死胡  同里,则必须返回重...
  • xxpresent
  • xxpresent
  • 2016-11-30 21:27
  • 2015
    个人资料
    • 访问:214775次
    • 积分:3937
    • 等级:
    • 排名:第9524名
    • 原创:96篇
    • 转载:77篇
    • 译文:4篇
    • 评论:198条
    最新评论