數(shù)據(jù)結(jié)構(gòu)課程設(shè)計---走迷宮游戲_第1頁
已閱讀1頁,還剩12頁未讀, 繼續(xù)免費閱讀

下載本文檔

版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請進行舉報或認領(lǐng)

文檔簡介

1、<p><b>  課程設(shè)計說明書</b></p><p> ?。〝?shù)據(jù)結(jié)構(gòu)課程設(shè)計)</p><p>  專 業(yè): </p><p>  課程名稱: 數(shù)據(jù)結(jié)構(gòu)課程設(shè)計 班級: </p><p>  設(shè)計題目:

2、 走迷宮游戲 </p><p>  設(shè)計時間: 2013-2-25 至 2013-3-8 </p><p>  評 語:_________________________________</p><p>  _________________________________________</p>&

3、lt;p>  _________________________________________</p><p>  _________________________________________</p><p>  _________________________________________</p><p>  評閱成績:__ __評閱教師:_

4、_ </p><p>  一、問題描述與需求分析</p><p><b>  1、問題描述</b></p><p>  程序開始運行時顯示一個迷宮地圖,迷宮中央有一只老鼠,迷宮的右下方有一個糧倉。游戲的任務(wù)是使用鍵盤上的方向鍵操縱老鼠在規(guī)定的時間內(nèi)走到糧倉處。</p><p><b>  要求:

5、</b></p><p>  老鼠形象可辨認,可用鍵盤操縱老鼠上下左右移動;</p><p>  迷宮的墻足夠結(jié)實,老鼠不能穿墻而過;</p><p>  正確檢測結(jié)果,若老鼠在規(guī)定時間內(nèi)走到糧倉處,提示成功,否則提示失??;</p><p>  添加編輯迷宮功能,可修改當前迷宮,修改內(nèi)容:墻變路、路變墻;</p>&

6、lt;p>  找出走出迷宮的所有路徑,以及最短路徑;</p><p>  6)利用序列化功能實現(xiàn)迷宮地圖文件的存盤和讀出等功能。</p><p><b>  功能需求分析</b></p><p>  老鼠形象設(shè)計,使用圖形化編程,繪制橢圓,填充顏色,繪制線。</p><p>  設(shè)置游戲者在探索過程中遇到迷宮邊界和

7、墻時,不可繼續(xù)前行,定義好迷宮邊界。</p><p>  使用time函數(shù)獲取系統(tǒng)時間,處理游戲所用時間,限定操作時間,對游戲者的位置有準確的判斷,當?shù)竭_出口時,可以識別,返回提示信息。</p><p>  對于迷宮的所有路徑的求解,比較最短路徑,最小生成樹算法。</p><p>  5.對迷宮的地圖可將其存儲到二進制文件中,在下次使用時直接調(diào)用,讀取文件。<

8、/p><p><b>  二、概要設(shè)計</b></p><p><b>  1、總體設(shè)計思路</b></p><p>  在程序中,采用二維數(shù)組存儲迷宮地圖(0:墻 1:路),在探索迷宮過程中采用棧的數(shù)據(jù)結(jié)構(gòu)存儲探索迷宮時的全部路徑和有效路徑,因棧的“后進先出”結(jié)構(gòu)非常適合探索過程中的退步,即可以保證在任何位置都可沿原路退回。

9、在探索迷宮過程中采用的是“窮舉求解”的方法,即從入口出發(fā),順某一方向向前探索,若能走通,則繼續(xù)往前走;否則沿原路退回,換一個方向在繼續(xù)探索,直到所有可能的通路都探索到為止。</p><p><b>  2、 模塊簡介</b></p><p>  程序由以下幾個模塊組成:</p><p>  迷宮地圖隨機生成模塊:</p><

10、p>  入口:int **Maze()</p><p>  出口:return maze;</p><p>  實現(xiàn)功能:該函數(shù)使用 new函數(shù)為指向二維數(shù)組的指針maze 申請存儲空間,分兩步實現(xiàn),先申請長度等于行數(shù)加2 的二級指針,然后為每個二維指針申請存儲空間。</p><p>  調(diào)用包含在頭文件 stdlib .h 中的庫函數(shù),隨機數(shù)生成器 sra

11、nd( ),rand( ), 及包含在頭文件time.h 中的系統(tǒng)時間函數(shù)time(),srand((unsigned)time(NULL)) 使用系統(tǒng)時間,傳入空指針NULL,作為初始化種子,使得在后面調(diào)用的rand()%2函數(shù)產(chǎn)生不同的隨機數(shù),取余之后為(0和1),從而實現(xiàn)了迷宮地圖的隨機生成,但使用該方法產(chǎn)生的迷宮地圖中不一定存在一條從入口到出口的路徑。</p><p>  最

12、后使用for循環(huán)嵌套給迷宮的上、下、左、右邊界賦值0(為墻壁);指定迷宮的入口與出口位置同時賦值為1(為通道)。</p><p>  因定義了二維指針類型函數(shù),故在其它函數(shù)調(diào)用該函數(shù)時返回指向迷宮地圖的二維指針,使得調(diào)用迷宮地圖極為方便。</p><p><b>  棧操作實現(xiàn)模塊:</b></p><p>  該模塊共包含:①初始化棧,②元素

13、入棧,③元素出棧,④刪除棧頂元素,⑤棧的遍歷 5個函數(shù)。</p><p><b> ?、俪跏蓟瘲?lt;/b></p><p>  入口:int StackTraverse(SqStack *s) </p><p>  出口:exit(OVERFLOW); return TRUE; </p><p>  實現(xiàn)功能:初始化

14、一個棧,并為其分配存儲空間初始量。形參為棧類型指針,調(diào)用時傳遞實參全局變量realPath Path 地址,調(diào)用包含在頭文件 malloc.h 中的庫函數(shù)malloc 根據(jù)棧的存儲空間初始量及SqStack所占字節(jié)為其動態(tài)分配內(nèi)存,最后,將內(nèi)存地址賦給棧底指針,同時使棧頂指針也指向該內(nèi)存,棧的大小為存儲空間初始量;當分配失敗時,返回空指針NULL,退出該函數(shù)同時輸出錯誤提醒語句,以便調(diào)試;分配成功,返回TRUE,表明該函數(shù)已執(zhí)行。<

15、;/p><p>  這樣做的優(yōu)點是節(jié)省了內(nèi)存,根據(jù)存儲使用量動態(tài)分配,在使用結(jié)束后可及時釋放該內(nèi)存。</p><p><b> ?、谠厝霔?lt;/b></p><p>  入口:int Push(SqStack *s,SElemType e)</p><p>  出口:exit(OVERFLOW); return TRUE;

16、</p><p>  實現(xiàn)功能:向棧中添加新元素。形參為棧類型指針,棧元素類型e ,調(diào)用時傳遞實參地址及入棧元素;</p><p>  但需注意的是:在入棧之前要判斷棧是否還有空間容納新元素,如棧滿(s->top - s->base >= s->stackSize)則應(yīng)使用 realloc 函數(shù)為棧分配存儲空間增量(STACKINCREMENT),在棧底指針base

17、 之后追加增量,同時將新的地址賦給base,此時應(yīng)重新定位棧頂指針(s->top = s->base + s->stacksize;),因重新分配了空間base值已改變,所以top值也就相應(yīng)的改變,才能指向新的元素;如存儲分配失敗,則輸出提醒語句,退出函數(shù);否則,新元素e 入棧,棧頂指針top++ ,返回TRUE ,函數(shù)正常退出。</p><p><b>  ③元素出棧</b&g

18、t;</p><p>  入口:SElemType GetTop(SqStack *s) </p><p>  出口:exit(ERROR); return *(s->top - 1);</p><p>  實現(xiàn)功能:若棧非空,獲取棧頂元素,并返回其值。因函數(shù)類型為棧元素類型,故可直接返回棧頂元素,需注意的是:在出棧之前要判斷棧是否非空(s->to

19、p == s->base),若base 值等于 top 值,則表明棧空,輸出錯誤提醒語句,強制退出函數(shù);否則,top 值減 1 ,返回棧頂元素,此時函數(shù)正常退出。</p><p><b>  ④刪除棧頂元素</b></p><p>  入口:int Pop(SqStack *s) </p><p>  出口: exit(ERROR);

20、return TRUE;</p><p>  實現(xiàn)功能:若棧非空,則刪除棧頂元素。同樣,在刪除元素之前需判斷棧是否非空,是,則棧頂指針 top - - 返回TRUE ,函數(shù)正常退出;否(s->top == s->base),則輸出錯誤提醒語句,強制退出函數(shù)。</p><p><b>  ⑤棧的遍歷</b></p><p>  入口:

21、int StackTraverse(SqStack *s)</p><p>  出口:return TRUE;</p><p>  實現(xiàn)功能:逆序遍歷棧,先指向棧底元素,以后依次向上增 1 ,輸出迷宮從入口到出口的路徑。當base 值小于top - 1 時執(zhí)行 while 循環(huán),由棧底依次向上輸出棧中元素,s -> base ++ ,不滿足條件時跳出 while 循環(huán),此時棧底指針

22、指向棧頂元素,輸出棧中最后一個元素,即迷宮通道中的出口位置。</p><p><b>  探索迷宮操作模塊:</b></p><p>  該模塊共包含:①判斷當前位置是否走過,②獲取東面相鄰位置,③獲取南面相鄰位置,④獲取西面相鄰位置,⑤獲取北面相鄰位置,⑥獲取下一可通行位置,以及⑦獲取迷宮路徑函數(shù) 7個函數(shù),其中獲取迷宮路徑函數(shù)為主要函數(shù),調(diào)用其他函數(shù)以實現(xiàn)獲取迷宮

23、路徑,并將其存儲到棧中。</p><p>  ①判斷當前位置是否走過</p><p>  入口:int UnPass(SqStack path, SElemType e);</p><p>  出口:return flag;</p><p>  實現(xiàn)功能:在探索迷宮時,調(diào)用該函數(shù),判斷當前位置是否走過。形參為棧類型 path ,棧元素類型 e

24、 ,在調(diào)用時傳遞全局變量Path ,即存儲探索迷宮時走過的所有路徑的棧,當前位置的棧元素類型;在該函數(shù)中,定義整型數(shù) flag = 1 作為標記,當棧Path 非空時執(zhí)行 while 循環(huán),比較當前位置對應(yīng)坐標是否與出棧元素的坐標相等,即判斷當前位置是否在Path 的路徑中出現(xiàn)過,若滿足條件,標記flag = 0 ,Path.top -- ,即每執(zhí)行一次循環(huán),頭指針下移一個位置,直到不滿足條件時跳出循環(huán),即將Path 中所有元素都與當前

25、位置作了比較;若有符合要求的,返回標記flag = 0 ,表明該位置走過;否則,返回標記flag=1,該位置未曾走過。</p><p><b> ?、讷@取東面相鄰位置</b></p><p>  入口:SElemType getEast(int **m,SElemType e)</p><p>  出口:return e;</p>

26、<p>  實現(xiàn)功能:獲取東面相鄰位置信息,返回棧元素類型,包含位置坐標,方向。形參為二維指針m ,棧元素類型 e ,調(diào)用時傳遞指向迷宮地圖的二維指針,及當前位置;</p><p>  注:以屏幕左上角為坐標原點,水平向右為 y 軸正方向,豎直向下為 x 軸正方向。</p><p>  判斷當前位置未到迷宮右(東)邊界時,當前位置 y 坐標加 1(e.curpos.y += 1

27、;),將東面相鄰位置在迷宮地圖中的值賦給當前位置(e.curpos.val = m[e.curpos.x][e.curpos.y];) ,返回 e ,從而實現(xiàn)了獲取當前位置的東面相鄰位置信息。</p><p><b> ?、郢@取南面相鄰位置</b></p><p>  入口:SElemType getSouth(int **m,SElemType e)</p&g

28、t;<p>  出口:return e;</p><p>  實現(xiàn)功能:獲取南面相鄰位置信息。需判斷當前位置是否已到迷宮下(南)邊界。</p><p>  x 位置坐標加 1(e.curpos.x += 1;)</p><p><b> ?、塬@取西面相鄰位置</b></p><p>  入口:SElemTy

29、pe getWest(int **m,SElemType e)</p><p>  出口:return e;</p><p>  實現(xiàn)功能:獲取西面相鄰位置信息。需判斷當前位置是否已到迷宮左(西)邊界。</p><p>  y 位置坐標加 1(e.curpos.y += 1;)</p><p><b> ?、莴@取北面相鄰位置<

30、/b></p><p>  入口:SElemType getNorth(int **m,SElemType e)</p><p>  出口:return e;</p><p>  實現(xiàn)功能:獲取北面相鄰位置信息。需判斷當前位置是否已到迷宮上(北)邊界。</p><p>  x 位置坐標減 1(e.curpos.x -= 1;)</

31、p><p> ?、瞢@取下一可通行位置</p><p>  入口:SElemType NextPos(SElemType e)</p><p>  出口:return next;</p><p>  實現(xiàn)功能:在當前位置,向四個方向(東、南、西、北)探索,調(diào)用②、③、④、⑤函數(shù),若相鄰位置可行走,且未曾走過,則返回該位置信息,將當前位置切換到下一位

32、置。</p><p><b>  ⑦獲取迷宮路徑函數(shù)</b></p><p>  入口:int MazePath(Position start,Position end);</p><p>  出口:return TRUE; return FALSE;</p><p>  實現(xiàn)功能:若迷宮Maze 中存在從入口 sta

33、rt 到出口 end 的通道,則求得一條存放在棧中realPath(從棧底到棧頂),并返回 TRUE ;否則返回 FALSE 。</p><p><b>  輔助函數(shù)模塊:</b></p><p><b> ?、佥敵雒詫m地圖</b></p><p>  入口:int PrintMap(int **temp);</p&

34、gt;<p>  出口:return TRUE;</p><p>  實現(xiàn)功能:在主函數(shù)中調(diào)用,傳遞實參指向迷宮地圖的二維指針,使用 for 循環(huán)嵌套輸出迷宮地圖。</p><p><b> ?、阱e誤消息提示</b></p><p>  入口:int errormessage()</p><p>  出口

35、:return TRUE;</p><p>  實現(xiàn)功能:錯誤消息提示</p><p><b>  主函數(shù)模塊:</b></p><p>  入口:int main() </p><p>  出口:return 0;</p><p>  實現(xiàn)功能:程序執(zhí)行的入口,在主函數(shù)中調(diào)用各個模塊,實現(xiàn)程序的

36、運行。</p><p>  初始化棧,Path 用于存放探索迷宮時的全部路徑,realPath 用于存放迷宮入口到出口的有效路徑;初始化游戲參數(shù),對全局變量map入口、出口位置坐標、對應(yīng)地圖值(為1)進行賦值;調(diào)用迷宮地圖輸出函數(shù),輸出迷宮;調(diào)用獲取迷宮路徑函數(shù),若存在一條路徑,則存放到棧realPath ;調(diào)用遍歷棧函數(shù),逆序輸出棧中元素(從棧底到棧頂),即從入口位置到出口位置的一條路徑。return 0; 主

37、函數(shù)結(jié)束,程序運行結(jié)束。</p><p><b>  三、詳細設(shè)計</b></p><p><b>  1、數(shù)據(jù)結(jié)構(gòu)設(shè)計</b></p><p> ?。?)程序中定義了迷宮的位置坐標,結(jié)構(gòu)體類型Position ,包含整型數(shù) x , y 存儲迷宮地圖二維數(shù)組對應(yīng)的行和列坐標,整型數(shù) val 用于存儲迷宮地圖的值,如0和1。

38、</p><p>  特別說明: val 值是作為迷宮探索時的重要判斷標記,若當前位置的四面為墻(0)或已走過,則在切換下一位置的函數(shù)NextPos中所返回val的值為-1。</p><p><b>  詳細定義如下:</b></p><p>  typedef struct </p>

39、;<p>  {int x; //二維數(shù)組maze下標 </p><p><b>  int y;</b></p><p>  int val; //maze[x][y]的值</p><p>  }Position;

40、 //游戲中的位置坐標</p><p>  程序中定義了結(jié)構(gòu)體類型MapCfg ,及對應(yīng)該結(jié)構(gòu)體類型全局變量map,包含上一結(jié)構(gòu)體類型Position變量start, end,即迷宮的起始位置與結(jié)束位置,在調(diào)用探索迷宮函數(shù)時,傳遞實參起始位置,結(jié)束位置為全局變量直接調(diào)用,從而使得在判斷結(jié)束位置時更加方便,但需注意的一點是:在調(diào)用之前要給起始、結(jié)束位置坐標變量賦

41、初值。</p><p><b>  詳細定義如下:</b></p><p>  typedef struct</p><p>  {Position start; //起始位置</p><p>  Position end; //結(jié)束位

42、置</p><p><b>  }MapCfg;</b></p><p>  MapCfg map;</p><p>  程序中定義了迷宮中路徑(單獨每一通道塊)所存儲的結(jié)構(gòu)體類型,包含當前位置坐標,及從該通道塊走向下一通道塊的“方向”,如:1:東,2:南,3:西,4:北。注:為存儲迷宮路徑的數(shù)據(jù)結(jié)構(gòu)棧的元素類型。</p><

43、;p><b>  詳細定義如下:</b></p><p>  typedef struct</p><p>  {Position curpos; //通道塊在迷宮中的位置坐標</p><p>  int di; //從此通道塊走向下一通道塊的"方向"&

44、lt;/p><p>  }SElemType;</p><p>  程序中定義了探索迷宮、和從入口 start 到出口 end 的通道,所存儲的數(shù)據(jù)結(jié)構(gòu)棧,為全局變量Path,realPath ,其存儲方式為順序棧,包含指向棧元素類型的棧底base和棧頂top指針,及棧的存儲空間stacksize ,需注意的是:在初始化棧時要分配給存儲空間初始量,在后續(xù)使用過程中可為其分配增量,以滿足存儲要求

45、。</p><p><b>  詳細定義如下:</b></p><p>  typedef struct{</p><p>  SElemType* base; //棧底指針</p><p>  SElemType* top; //棧頂指針&l

46、t;/p><p>  int stacksize; //棧的大小</p><p><b>  }SqStack;</b></p><p>  SqStack realPath; //存放路徑</p><p>  SqStack P

47、ath; //探索迷宮</p><p>  2、 算法分析與實現(xiàn)</p><p>  主要功能模塊的算法設(shè)計分析。</p><p>  int MazePath(Position start,Position end)</p><p>  { int **temp = Maze();</p>

48、;<p>  SElemType e;</p><p>  e.curpos.x = map.start.x; //設(shè)定"當前位置"為"入口位置"</p><p>  e.curpos.y = map.start.y;</p><p>  e.curpos.val = temp[m

49、ap.start.x][map.start.y];</p><p><b>  do</b></p><p>  {if( UnPass(Path, e) ) //當前位置可以通過,即是未曾走過的通道塊</p><p><b>  {</b></p><p>  Push(&real

50、Path,e);</p><p>  Push(&Path,e);</p><p>  e = NextPos(e);</p><p>  if(e.curpos.x == map.end.x && e.curpos.y == map.end.y) </p><p>  {

51、//到達終點(出口)</p><p>  Push(&realPath,e);</p><p>  Push(&Path,e);</p><p>  StackTraverse(&Path);</p><p>  return TRUE;</p><p><b>  }</

52、b></p><p>  else if(e.curpos.val == -1)</p><p>  {//當前位置的四面或為墻或已走過</p><p>  Pop(&realPath);//刪除真實路徑的棧頂元素</p><p>  e = GetTop(&realPath);//令e指向棧頂

53、元素</p><p><b>  }</b></p><p><b>  }</b></p><p><b>  else</b></p><p>  {//如果當前位置已經(jīng)走過,說明原來測試的方向不對,現(xiàn)在嘗試其它方向 </p><p&g

54、t;  e = NextPos(e);</p><p>  if (e.curpos.val == -1)</p><p>  {//仍不通,刪除真實路徑的棧頂元素</p><p>  Pop(&realPath);</p><p>  e = GetTop(&realPath);//令cur指向棧頂元素</p&

55、gt;<p><b>  }</b></p><p><b>  }</b></p><p>  }while(e.curpos.x != end.x || e.curpos.y != end.y);</p><p>  return FALSE;</p><p><b> 

56、 }</b></p><p>  四、運行結(jié)果和調(diào)試分析</p><p><b>  程序運行結(jié)果圖:</b></p><p>  1.計算機“窮舉求解法”,輸出迷宮入口到出口路徑。</p><p>  2.用二維數(shù)組表示迷宮地圖,0 為墻,1 為道路。</p><p>  在程序調(diào)試

57、過程中遇到變量不能被賦值的問題,后來經(jīng)過按F5進行斷點調(diào)試,根據(jù)錯誤提醒,發(fā)現(xiàn)了問題所在,得到了及時的修改。在本程序中對于迷宮當前位置是否走過的判斷,需要對存儲所有路徑的realPath 一一訪問,對比當前位置是否被訪問過,追其原因,是由于探索過程中并沒有做好該位置已被訪問的標記,導(dǎo)致每一位置都需重新定位,進行判斷,浪費了時間與存儲空間。</p><p>  改進:在棧元素類型,即迷宮中每一位置,增加flag 標

58、記,若已訪問則標記 -1 ,下次獲得該位置坐標后便可得知該位置是否被訪問過的信息。</p><p><b>  五、總結(jié)體會</b></p><p>  在這次數(shù)據(jù)結(jié)構(gòu)課程設(shè)計中,我并沒有很好的完成所選題目的要求,如圖形化的編程,以及windows 窗口的編程,還有對于鍵盤操縱游戲者。在設(shè)計初期,從網(wǎng)上查閱了很多資料,對于C語言圖形化編程有了一定的了解,但是由于之前并

59、沒有接觸過,理解不夠深刻,還不能做到熟練應(yīng)用,就自己現(xiàn)在了解到的圖形化編程,有兩個選擇,windows編程和EasyX吧這個網(wǎng)站下載grapics.h頭文件和Lib文件,有很多繪圖函數(shù)可用,至于windows編程,感覺稍有難度,但能做出高質(zhì)量的程序,由于我對這兩種編程都不是很熟練,最終還是選擇了win32 console Application ,而且覺得核心與重點在算法與數(shù)據(jù)結(jié)構(gòu)的設(shè)計,覺得自己的選擇是正確的,通過這次編程,對程序設(shè)計

60、有了更多的自己的理解,在設(shè)計程序之前的工作,都是非常重要的,因為作為編程者要清楚的知道自己在做什么,思路清晰才會順利,而且還有一個問題就是C語言和數(shù)據(jù)結(jié)構(gòu)的基礎(chǔ)很重要,對一些概念往往并不是做到了完全的理解,以至于在編程時錯誤百出,覺得很有必要再重新學習一次C語言和數(shù)據(jù)結(jié)構(gòu)課本,同時也認識到了這兩門課程的重要性。在這次課程設(shè)計中,最大的收獲就是</p><p>  最后,雖然此次課程設(shè)計結(jié)束了,但并不意味著,程序設(shè)

61、計結(jié)束,甚至是一個很好的開始,希望自己再接再勵,將程序的每一個功能完善,做到更好!</p><p><b>  參考文獻</b></p><p><b>  如:</b></p><p>  [1]嚴蔚敏,吳偉民編著.《數(shù)據(jù)結(jié)構(gòu)》(C語言版).清華大學出版社。</p><p>  [2]譚浩強編著.

溫馨提示

  • 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
  • 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
  • 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會有圖紙預(yù)覽,若沒有圖紙預(yù)覽就沒有圖紙。
  • 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
  • 5. 眾賞文庫僅提供信息存儲空間,僅對用戶上傳內(nèi)容的表現(xiàn)方式做保護處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負責。
  • 6. 下載文件中如有侵權(quán)或不適當內(nèi)容,請與我們聯(lián)系,我們立即糾正。
  • 7. 本站不保證下載資源的準確性、安全性和完整性, 同時也不承擔用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。

評論

0/150

提交評論