當前位置:編程學習大全網 - 編程語言 - 單機版的五子棋程序的算法是什麽哦!

單機版的五子棋程序的算法是什麽哦!

五子棋是壹種受大眾廣泛喜愛的遊戲,其規則簡單,變化多端,非常富有趣味性和消遣性。這裏設計和實現了壹個人機對下的五子棋程序,采用了博弈樹的方法,應用了剪枝和最大最小樹原理進行搜索發現最好的下子位置。介紹五子棋程序的數據結構、評分規則、勝負判斷方法和搜索算法過程。

壹、相關的數據結構

關於盤面情況的表示,以鏈表形式表示當前盤面的情況,目的是可以允許用戶進行悔棋、回退等操作。

CList StepList;

其中Step結構的表示為:

struct Step

{

int m; //m,n表示兩個坐標值

int n;

char side; //side表示下子方

};

以數組形式保存當前盤面的情況,

目的是為了在顯示當前盤面情況時使用:

char FiveArea[FIVE_MAX_LINE][FIVE_MAX_LINE];

其中FIVE_MAX_LINE表示盤面最大的行數。

同時由於需要在遞歸搜索的過程中考慮時間和空間有效性,只找出就當前情況來說相對比較好的幾個盤面,而不是對所有的可下子的位置都進行搜索,這裏用變量CountList來表示當前搜索中可以選擇的所有新的盤面情況對象的集合:

CList CountList;

其中類CBoardSituiton為:

class CBoardSituation

{

CList StepList; //每壹步的列表

char FiveArea[FIVE_MAX_LINE][FIVE_MAX_LINE];

struct Step machineStep; //機器所下的那壹步

double value; //該種盤面狀態所得到的分數

}

二、評分規則

對於下子的重要性評分,需要從六個位置來考慮當前棋局的情況,分別為:-,?,/,\,//,\\

實際上需要考慮在這六個位置上某壹方所形成的子的布局的情況,對於在還沒有子的地方落子以後的當前局面的評分,主要是為了說明在這個地方下子的重要性程度,設定了壹個簡單的規則來表示當前棋面對機器方的分數。

基本的規則如下:

判斷是否能成5, 如果是機器方的話給予100000分,如果是人方的話給予-100000 分;

判斷是否能成活4或者是雙死4或者是死4活3,如果是機器方的話給予10000分,如果是人方的話給予-10000分;

判斷是否已成雙活3,如果是機器方的話給予5000分,如果是人方的話給予-5000 分;

判斷是否成死3活3,如果是機器方的話給予1000分,如果是人方的話給予-1000 分;

判斷是否能成死4,如果是機器方的話給予500分,如果是人方的話給予-500分;

判斷是否能成單活3,如果是機器方的話給予200分,如果是人方的話給予-200分;

判斷是否已成雙活2,如果是機器方的話給予100分,如果是人方的話給予-100分;

判斷是否能成死3,如果是機器方的話給予50分,如果是人方的話給予-50分;

判斷是否能成雙活2,如果是機器方的話給予10分,如果是人方的話給予-10分;

判斷是否能成活2,如果是機器方的話給予5分,如果是人方的話給予-5分;

判斷是否能成死2,如果是機器方的話給予3分,如果是人方的話給予-3分。

實際上對當前的局面按照上面的規則的順序進行比較,如果滿足某壹條規則的話,就給該局面打分並保存,然後退出規則的匹配。註意這裏的規則是根據壹般的下棋規律的壹個總結,在實際運行的時候,用戶可以添加規則和對評分機制加以修正。

三、勝負判斷

實際上,是根據當前最後壹個落子的情況來判斷勝負的。實際上需要從四個位置判斷,以該子為出發點的水平,豎直和兩條分別為 45度角和135度角的線,目的是看在這四個方向是否最後落子的壹方構成連續五個的棋子,如果是的話,就表示該盤棋局已經分出勝負。具體見下面的圖示:

四、搜索算法實現描述

註意下面的核心的算法中的變量currentBoardSituation,表示當前機器最新的盤面情況, CountList表示第壹層子節點可以選擇的較好的盤面的集合。核心的算法如下:

void MainDealFunction()

{

value=-MAXINT; //對初始根節點的value賦值

CalSeveralGoodPlace(currentBoardSituation,CountList);

//該函數是根據當前的盤面情況來比較得到比較好的可以考慮的幾個盤面的情況,可以根據實際的得分情況選取分數比較高的幾個盤面,也就是說在第壹層節點選擇的時候采用貪婪算法,直接找出相對分數比較高的幾個形成第壹層節點,目的是為了提高搜索速度和防止堆棧溢出。

pos=CountList.GetHeadPosition();

CBoardSituation* pBoard;

for(i=0;ivalue=Search(pBoard,min,value,0);

Value=Select(value,pBoard->value,max);

//取value和pBoard->value中大的賦給根節點

}

for(i=0;ivalue)

//找出那壹個得到最高分的盤面

{

currentBoardSituation=pBoard;

PlayerMode=min; //當前下子方改為人

Break;

}

}

其中對於Search函數的表示如下:實際上核心的算法是壹個剪枝過程,其中在這個搜索過程中相關的四個參數為:(1)當前棋局情況;(2)當前的下子方,可以是機器(max)或者是人(min);(3)父節點的值oldValue;(4)當前的搜索深度depth。

double Search(CBoardSituation&

board,int mode,double oldvalue,int depth)

{

CList m_DeepList;

if(deptholdvalue))== TRUE)

{

if(mode==max)

value=select(value,search(successor

Board,min,value,depth+1),max);

else

&nb

sp; value=select(value,search(successor

Board,max,value,depth+1),min);

}

return value;

}

else

{

if ( goal(board)<>0)

//這裏goal(board)<>0表示已經可以分出勝負

return goal(board);

else

return evlation(board);

}

}

註意這裏的goal(board)函數是用來判斷當前盤面是否可以分出勝負,而evlation(board)是對當前的盤面從機器的角度進行打分。

下面是Select函數的介紹,這個函數的主要目的是根據 PlayerMode情況,即是機器還是用戶來返回節點的應有的值。

double Select(double a,double b,int mode)

{

if(a>b && mode==max)? (a< b && mode==min)

return a;

else

return b;

}

五、小結

在Windows操作系統下,用VC++實現了這個人機對戰的五子棋程序。和國內許多只是采用規則或者只是采用簡單遞歸而沒有剪枝的那些程序相比,在智力上和時間有效性上都要好於這些程序。同時所討論的方法和設計過程為用戶設計其他的遊戲(如象棋和圍棋等)提供了壹個參考。

  • 上一篇:在線等!急! 用PHP編寫程序,實現簡單的用戶登錄頁面 (1)制作login.html用戶登錄頁面,效果圖如下:
  • 下一篇:我的世界手機聯機地圖加載不全怎麽辦?
  • copyright 2024編程學習大全網