利用SDK实现迷宫算法

2016-01-29 12:15 7 1 收藏

利用SDK实现迷宫算法,利用SDK实现迷宫算法

【 tulaoshi.com - C语言心得技巧 】

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

作者:赖锋

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


下载本文示例源代码

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



迷宫算法还是老路子,回溯法和堆栈实现,我采用的是堆栈实现. 使用双向链表摸拟堆栈,使用一个 ptrFirst 和一个 ptrLast 和作为堆栈的栈底和栈顶指针, 定义一 个堆栈元素结构, 这个结构保存迷宫中的位置.

typedef struct _tagNode {intnRow;intnColumn;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 消息进行内存释放.

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

附带的源代码中有两个贴图的小程序,写得尽量简单,对刚学

来源:http://www.tulaoshi.com/n/20160129/1485241.html

延伸阅读
代码如下: import java.util.ArrayList;  import java.util.Collections;  import java.util.List;  /**   * Created with IntelliJ IDEA.   * User: dell   * Date: 13-3-5   * Time: 下午8:38   * To change this template use File | Settings | File Templates.   */  public class G...
轻松实现DES算法查看器 作者:小帅 下载源代码 DES(Data Encrypton Standard) 算法的实现网上已经有很多,本人在此讲述的是在DES算法加密过程中如何查看16迭代过程中生成的Ki,Li,Ri,Fi,Si等,这样可以当做一个DES加密对照器,这样可以方便的发现你在加密过程中出现的错误! 图一:程序运行界面 本程序用...
在MySQL 中,只有一种 Join 算法,就是大名鼎鼎的 Nested Loop Join,他没有其他很多数据库所提供的 Hash Join,也没有 Sort Merge Join。顾名思义,Nested Loop Join 实际上就是通过驱动表的结果集作为循环基础数据,然后一条一条的通过该结果集中的数据作为过滤条件到下一个表中查询数据,然后合并结果。如果还有第三个参与 Join,则再通...
代码如下: public class MirrorView extends View {  Paint m_paint;  int m_nShadowH;  Drawable m_dw;  Bitmap m_bitmap;  //xxxx  Matrix mMatrix;  int shadowHeight;  public MirrorView(Context context, Bitmap bitmap) {   super(context);   m_bitmap = bitmap;   _Init(); ...
参数model有一个二维数组data,及阶数matrix // .h文件@class DataModel; @interface Algorithm : NSObject @property (nonatomic,assign) int addScore; // 加分 - (void)caculateTop:(DataModel *)model; // 上滑规则- (void)caculateBottom:(DataModel *)model; // 下滑规则- (void)caculateLeft:(DataModel *)model; // 左滑规则- (void...

经验教程

379

收藏

74
微博分享 QQ分享 QQ空间 手机页面 收藏网站 回到头部