圖的遍歷迷宮算法淺析_第1頁(yè)
圖的遍歷迷宮算法淺析_第2頁(yè)
圖的遍歷迷宮算法淺析_第3頁(yè)
圖的遍歷迷宮算法淺析_第4頁(yè)
圖的遍歷迷宮算法淺析_第5頁(yè)
已閱讀5頁(yè),還剩19頁(yè)未讀, 繼續(xù)免費(fèi)閱讀

下載本文檔

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

文檔簡(jiǎn)介

1、圖的遍歷迷宮算法QQ:513696765作者:小牛1. 引言 在平常的游戲中,我們常常會(huì)碰到隨機(jī)生成的地圖。這里我們就來(lái)看看一個(gè)簡(jiǎn)單的隨機(jī)迷宮是如何生成。2. 迷宮描述 隨機(jī)生成一個(gè)m * n的迷宮,可用一個(gè)矩陣mazemn來(lái)表示,如圖: 圖1.1 圖1.2這里是兩個(gè)迷宮的例子,其中“”表示障礙物(Obstacle block)。以圖1.1迷宮為例,我們可用一個(gè)9 * 9的矩陣來(lái)表示:1 1 1 1 1 1 1 1 10 0 0 0 0 0 0 0 11 1 1 1 1 1 1 0 11 0 0 0 0 0 1 0 11 0 1 1 1 1 1 0 11 0 1 0 0 0 0 0 11 0

2、1 0 1 1 1 1 11 0 0 0 0 0 0 0 01 1 1 1 1 1 1 1 1(矩陣中1表示是障礙物,0表示可以行走)3、迷宮生成算法(藍(lán)色表示可以行走,棕色表示是墻壁) 圖3.1如圖3.1所示為迷宮的初始化情形,迷宮如果除去其迷宮的外圍框架就是一個(gè)7*7的矩陣,如果要生成一個(gè)完整的迷宮,那么就要遍歷完圖3.1所示的每一個(gè)可以行走的點(diǎn),其中遍歷的點(diǎn)也包含了入口和出口,如果每一個(gè)點(diǎn)都遍歷完了就會(huì)生成一棵完整的遍歷樹,這棵遍歷樹包含了入口和出口所以這顆樹所描述的迷宮是有解的(即:入口到出口時(shí)連通的。)圖3.2就表示圖1.1迷宮遍歷所得到的遍歷樹,樹包含了入口和出口所以此迷宮有解。

3、圖3.2圖3.3圖3.3表示的是圖3.2進(jìn)一步的處理后得到的最終迷宮圖,就是在圖3.2的基礎(chǔ)上把遍歷樹上的是墻壁的元素置為可行走的元素。3.1元素的遍歷圖3.1.1描述了各個(gè)遍歷點(diǎn)的連接關(guān)系其中1元素為起始遍歷點(diǎn),49元素為終點(diǎn)遍歷點(diǎn)我們聲明一個(gè)mazepoint類用來(lái)描述每一個(gè)遍歷點(diǎn)xtemp表示當(dāng)前點(diǎn)在數(shù)組中的x坐標(biāo),ytemp表示當(dāng)前點(diǎn)在數(shù)組中的y坐標(biāo),定義為mazepoint對(duì)象的next用來(lái)鏈表一個(gè)點(diǎn)的地址last用來(lái)鏈表上一個(gè)地址。圖3.1.1聲明了兩個(gè)mazepoint的變量head和tail用于存儲(chǔ)迷宮的鏈表的頭地址和尾地址由于在迷宮剛剛開始的時(shí)候初始化元素是第一個(gè)所以頭尾是相

4、同的在主函數(shù)中把head賦值給了p1,(p1是我們聲明的一個(gè)臨時(shí)存儲(chǔ)的變量;p1和p2用于新的鏈表元素生成)如圖3.1.2所示元素1它連接到元素3和元素15,所以它可以遍歷這兩個(gè)元素,假設(shè)遍歷的方向是隨機(jī)的,如果第一次現(xiàn)在向右遍歷那么元素3就被遍歷并把它標(biāo)志為flag(flag表示已經(jīng)遍歷過(guò)了不容許再次遍歷),所以元素1和元素3都標(biāo)志位flag說(shuō)明在其它元素想遍歷它們的時(shí)候是不能再次被遍歷了。這就有可能會(huì)遍歷成一棵完整的樹。 圖3.1.2如圖3.1.3所示圖1.1迷宮在生成時(shí)前13步,據(jù)圖我們可知在第13步的時(shí)候遍歷到元素19,但是元素19周圍的點(diǎn)都已經(jīng)遍歷過(guò)但同時(shí)還有其它元素還沒有遍歷到,所

5、以要后退到上一個(gè)遍歷過(guò)的元素判斷是否有遍歷的方向,所以鏈表到元素17,但是此元素也沒有可遍歷的方向所以要一直往上鏈表直到鏈表到某一個(gè)可以重新遍歷的點(diǎn)為之。 圖3.1.3圖3.1.4是往上鏈表的示意圖直到鏈表到元素45發(fā)現(xiàn)此元素?fù)碛锌杀闅v的方向,所以又重新往下建立新的鏈表,即元素47 圖3.1.4圖3.1.5是最終遍歷完后的遍歷路徑,此時(shí)只要沿著遍歷的方向把相應(yīng)的“墻”拆掉就可以生成一個(gè)無(wú)外框的迷宮并且只能生成奇數(shù)行和奇數(shù)列的迷宮。 圖3.1.5圖3.1.6是程序生成得33*33的迷宮 圖3.1.64、編程思路結(jié) 束屏幕輸出迷宮鏈接到上一個(gè)點(diǎn)tail=tail->last下一個(gè)點(diǎn)是否有方向

6、可遍歷把p1的地址賦給p2,同時(shí)把當(dāng)前點(diǎn)的x,y坐標(biāo)賦給p1,新建一個(gè)鏈表元素并且把其地址賦給p1,把p2的next連接到p1的地址,p1的last鏈接到p2,p1的next鏈接到NULL,這樣就生成了一個(gè)雙向的動(dòng)態(tài)鏈表每一個(gè)點(diǎn)是否遍歷完 即:pointcount=0?聲明和初始化mazepoint變量P1,p2,head,teil初始化p1變量成員變量并且把p1的地址傳遞給head開 始在可以遍歷的方向中隨機(jī)選擇一個(gè)方向遍歷并且把選擇的元素標(biāo)志為flag同時(shí)把x,y的坐標(biāo)更新到和當(dāng)前點(diǎn)的坐標(biāo),說(shuō)明當(dāng)前點(diǎn)不可再一次遍歷了pointcount-1說(shuō)明有一個(gè)點(diǎn)已經(jīng)遍歷過(guò)了,再把tail賦值為p1的

7、地址5、源代碼#include<iostream>#include <time.h>using std:cout;using std:cin;using std:endl;#define Row 33/用于定義迷宮的行必須為奇數(shù),否則不能遍歷到所以的點(diǎn)#define Col 33/用于定義迷宮的列必須為奇數(shù),否則不能遍歷到所以的點(diǎn)#define flag 1/點(diǎn)的標(biāo)志位,如果找到符合條件的點(diǎn)就把它置為flag,只要某一個(gè)點(diǎn)置位為flag就不能再次被訪問(wèn);int x_row=0;/迷宮的起始點(diǎn)位置的x坐標(biāo)int y_col=0;/迷宮的起始點(diǎn)位置的y坐標(biāo)int Left=

8、0;/初始化向左的方向標(biāo)志位Leftint Right=0;/初始化向右的方向標(biāo)志位Rightint Up=0;/初始化向上的方向標(biāo)志位Upint Dwon=0;/初始化向下的方向標(biāo)志位Dwonint pointcount=(Row+1)*(Col+1)/4-1);/初始化要遍歷的點(diǎn)數(shù),(因?yàn)榈谝粋€(gè)點(diǎn)不要遍歷了,所以總點(diǎn)數(shù)要減去1)int get_count();/定義可以獲得某一個(gè)點(diǎn)可以行走的方向數(shù)的函數(shù)class mazepoint/定義一個(gè)類用于存放每一個(gè)遍歷點(diǎn)的坐標(biāo)public:int xtemp; /用于存放一個(gè)遍歷點(diǎn)的x坐標(biāo)int ytemp;/用于存放一個(gè)遍歷點(diǎn)的y坐標(biāo)mazep

9、oint *next;/用于存放一個(gè)遍歷點(diǎn)的下一個(gè)遍歷點(diǎn)的地址mazepoint *last;/用于存放一個(gè)遍歷點(diǎn)的上一個(gè)遍歷點(diǎn)的地址;mazepoint *p1;mazepoint *p2;mazepoint *head=NULL;/初始化頭由于沒有指向如何位置所以賦值為NULLmazepoint *tail=NULL;/初始化尾由于沒有指向如何位置所以賦值為NULLint mazemapRowCol=/初始化迷宮地圖數(shù)組1;int main()p1=new mazepoint;/聲明一片內(nèi)存空間p1->xtemp=0;/初始化類元素x坐標(biāo)p1->ytemp=0;/初始化類元素y

10、坐標(biāo)p1->last=NULL;/初始類只有一個(gè)元素沒有前一個(gè)元素,鏈接到別的元素所以它的的上一個(gè)遍歷點(diǎn)位空NULLhead=p1;/把鏈表的頭地址賦值給p1tail=p1;/剛剛開始沒有其它元素所以把鏈表的尾地址也是p1int mazenum=0;/用于統(tǒng)計(jì)遍歷的次數(shù)srand(time(0);/用當(dāng)前時(shí)間做隨機(jī)種子數(shù)while(pointcount)/如果沒有遍歷完所有的點(diǎn)就繼續(xù)遍歷,以確保每一次的隨機(jī)數(shù)都不一樣int rdNo=rand();/獲得隨機(jī)數(shù)int Randtemp;/定義一個(gè)變量,用于存儲(chǔ)處理后的隨機(jī)數(shù)int tempflag;/定義一個(gè)變量,用于存儲(chǔ)某一個(gè)點(diǎn)可以行走

11、的方向數(shù)tempflag=get_count();/獲得某一個(gè)點(diǎn)可以行走的方向數(shù)if(tempflag!=0) /如果某一個(gè)點(diǎn)可以行走就處理隨機(jī)數(shù)然后隨機(jī)選擇行走的方向Randtemp=rdNo%tempflag;/根據(jù)返回的可以行走的方向提供行走方向數(shù)elseRandtemp=NULL;/如果不能行走就賦值為NULLcout<<"電腦隨機(jī)數(shù)為:" << rdNo<<endl;/輸出電腦產(chǎn)生的隨機(jī)數(shù)Randtemp+=1;/加1是為了和定義的方向Left,Right,Up,Dwon相匹配cout<<"隨機(jī)數(shù)為:&qu

12、ot; << Randtemp<<endl;/輸出處理后的隨機(jī)數(shù)if(tempflag!=0)/如果可以行走p2=p1;p1->xtemp=x_row;/保存當(dāng)前點(diǎn)的x坐標(biāo)p1->ytemp=y_col;/保存當(dāng)前點(diǎn)的y坐標(biāo)p1=new mazepoint;/聲明內(nèi)存p2->next=p1;/前一個(gè)遍歷點(diǎn)鏈表到下一個(gè)點(diǎn)的地址p1->last=p2;/后一個(gè)遍歷點(diǎn)鏈表到上一個(gè)點(diǎn)的地址p1->next=NULL;/新生產(chǎn)的點(diǎn)沒有鏈表到下一個(gè)點(diǎn)所以設(shè)為NULL。(新產(chǎn)生的點(diǎn)是鏈表的最后一個(gè)點(diǎn))if(Randtemp=Left &&

13、 mazemapx_rowy_col-2!=flag)/判斷左邊是否可以走如果可以走就不為;flagRandtemp=Left表示隨機(jī)數(shù)等于Left/*/mazemapx_rowy_col;新遍歷到的位置*/上一次確定的位置*/ mazemapx_rowy_col+1;新遍歷到位置的相同方向的臨近元素*/*y_col-=2;/如果可以左走就要把對(duì)應(yīng)的坐標(biāo)也要改變;也就是行值減2mazemapx_rowy_col=flag;/把遍歷到的新的迷宮地圖元素也要置flagmazemapx_rowy_col+1=flag;/把遍歷到的新的迷宮地圖元素相同方向的臨近元素也要置flag-pointcount

14、;/遍歷到新的一個(gè)點(diǎn),那么遍歷點(diǎn)的總數(shù)pointcount減1cout<<"I choice Left"<<endl;else if(Randtemp=Right && mazemapx_rowy_col+2!=flag)/判斷右邊是否可以走如果可以走就不為;flagRandtemp=Right表示隨機(jī)數(shù)等于Right/*/ mazemapx_rowy_col;新遍歷到的位置*/上一次確定的位置*/mazemapx_rowy_col-1;新遍歷到位置的相同方向的臨近元素*/*y_col+=2;/如果可以右走就要把對(duì)應(yīng)的坐標(biāo)也要改變;也

15、就是行值加2mazemapx_rowy_col=flag;/把遍歷到的新的迷宮地圖元素也要置flagmazemapx_rowy_col-1=flag;/把遍歷到的新的迷宮地圖元素相同方向的臨近元素也要置flag-pointcount;/遍歷到新的一個(gè)點(diǎn),那么遍歷點(diǎn)的總數(shù)pointcount減1cout<<"I choice Right"<<endl;else if(Randtemp=Up && mazemapx_row-2y_col!=flag)/判斷上邊是否可以走如果可以走就不為;flagRandtemp=Up表示隨機(jī)數(shù)等于Up/*

16、/ mazemapx_rowy_col;新遍歷到的位置*/ mazemapx_row+1y_col;新遍歷到位置的相同方向的臨近元素*/上一次確定的位置*/*x_row-=2;/如果可以上走就要把對(duì)應(yīng)的坐標(biāo)也要改變;也就是列值減2mazemapx_rowy_col=flag;/把遍歷到的新的迷宮地圖元素也要置flagmazemapx_row+1y_col=flag;/把遍歷到的新的迷宮地圖元素相同方向的臨近元素也要置flag-pointcount;/遍歷到新的一個(gè)點(diǎn),那么遍歷點(diǎn)的總數(shù)pointcount減1cout<<"I choice Up"<<e

17、ndl;else if(Randtemp=Dwon && mazemapx_row+2y_col!=flag)/判斷下邊是否可以走如果可以走就不為;flagRandtemp=Dwon表示隨機(jī)數(shù)等于Dwon/*/上一次確定的位置*/ mazemapx_row-1y_col;新遍歷到位置的相同方向的臨近元素*/ mazemapx_rowy_col;新遍歷到的位置*/*x_row+=2;/如果可以下走就要把對(duì)應(yīng)的坐標(biāo)也要改變;也就是列值加2mazemapx_rowy_col=flag;/把遍歷到的新的迷宮地圖元素也要置flagmazemapx_row-1y_col=flag;/把遍歷

18、到的新的迷宮地圖元素相同方向的臨近元素也要置flag-pointcount;/遍歷到新的一個(gè)點(diǎn),那么遍歷點(diǎn)的總數(shù)pointcount減1cout<<"I choice Dwon"<<endl;tail=p1;/把鏈表的尾巴賦值給新產(chǎn)生的點(diǎn)。(新產(chǎn)生的點(diǎn)是鏈表的最后一個(gè)點(diǎn))else/如果不可以行走就鏈表到上一個(gè)點(diǎn)直到鏈表到的某一個(gè)點(diǎn)可以行走為止cout<<"Pop To Last Point."<<endl;if(tail=head)/如果是第一個(gè)遍歷點(diǎn)那么鏈表的頭和尾巴重合tail=head;return

19、0;elsetail=tail->last;/鏈表到上一個(gè)點(diǎn)地址if(tail->last =NULL)tail=head;x_row=tail->xtemp;/讀取上一個(gè)遍歷點(diǎn)的x坐標(biāo)y_col=tail->ytemp;/讀取上一個(gè)遍歷點(diǎn)的y坐標(biāo)+mazenum;/執(zhí)行次數(shù)加1cout<<"Step is:"<<" "<<mazenum<<endl;/統(tǒng)計(jì)執(zhí)行的步數(shù)cout<<"Last Point coordinate is"<<&qu

20、ot; x is:"<<tail->xtemp<<" y is:"<<tail->ytemp<<endl;/輸出上一個(gè)點(diǎn)的X坐標(biāo)和Y坐標(biāo)cout<<"Now Point coordinate is"<<" x is:"<<x_row<<" y is:"<<y_col<<endl;/輸出當(dāng)前點(diǎn)的X坐標(biāo)和Y坐標(biāo)if(pointcount=0)/如果每一個(gè)遍歷點(diǎn)都遍歷到了就打印地圖

21、,生成得迷宮是沒有邊界,即:沒有外圍墻的迷宮for(int C=0;C<Col+2;+C)/輸出迷宮最上面的的圍墻即:上邊界cout<<""cout<<endl;for(int R=0;R<Row;+R)if(R=0)/入口不能輸出圍墻cout<<"IN" /打印入口elsecout<<""/輸出迷宮最左面的的圍墻即:左邊界for(int C=0;C<Col;+C)/打印各迷宮數(shù)組的元素if(mazemapRC=flag)/如果迷宮數(shù)組的元素是路的話就打印空白cout&

22、lt;<" "else /如果迷宮數(shù)組的元素是墻壁(flag)的話就打印墻cout<<""if(R=Row-1)/打印出口cout<<"OUT"elsecout<<""/輸出迷宮最右面的的圍墻即:右邊界cout<<endl;for( C=0;C<Col+2;+C)/輸出迷宮最下面的的圍墻即:下邊界cout<<""cout<<endl;return 0;int get_count()int count=0;/初始化

23、用于某一個(gè)點(diǎn)可以行走的方向數(shù),(count用于統(tǒng)計(jì)某一個(gè)點(diǎn)可以行走的方向數(shù))count=0;if(y_col-2<0)/判斷往左是否越界;即:是否超出迷宮數(shù)組if(y_col<0)y_col=0;Left=0;cout<<"Left not is Direction "<<"count is:"<<count<<endl;/輸出往左不能走elseif(mazemapx_rowy_col-2!=flag)/判斷往左是否可以走+count;Left=count;elsecout<<&qu

24、ot;Left not is Direction "<<"count is:"<<count<<endl;/輸出往左不能走if(y_col+2>Col)/判斷往右是否越界;即:是否超出迷宮數(shù)組if(y_col>Col)y_col=Col;Right=0;cout<<"Right not is Direction "<<"count is:"<<count<<endl;/輸出往右不能走elseif(mazemapx_rowy_col+2!=flag)/判斷往右是否可以走+count;Right=count;elsecout<<&quo

溫馨提示

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

最新文檔

評(píng)論

0/150

提交評(píng)論