|
|
五子棋是一种受大众广泛喜爱的游戏,其规则简单,变化多端,非常富有趣味性和消遣性。这里设计和实现了一个人机对下的五子棋程序,采用了博弈树的方法,应用了剪枝和最大最小树原理进行搜索发现最好的下子位置。介绍五子棋程序的数据结构、评分规则、胜负判断方法和搜索算法过程。 B) Y: N/ C C9 O. k4 u: y8 J/ P
% X/ S5 W/ v& W2 m- `
一、相关的数据结构
; u4 O3 d* Q. K6 g! l; D 关于盘面情况的表示,以链表形式表示当前盘面的情况,目的是可以允许用户进行悔棋、回退等操作。 ! ~ ^* i( F" B
CList StepList;
. @7 Q) f1 p. m 其中Step结构的表示为: 3 g) b+ ?7 j9 z- `) ^) Z+ b8 }, e& A' r
P* E" d+ f2 {0 Q+ R struct Step 7 C% [6 C# A" X/ b9 X$ D
{ * Q5 ]& i4 R3 H! o' w, T+ S
int m; //m,n表示两个坐标值 / m1 ]+ d0 T& u: h. J% N: q
int n; 6 S9 m4 p4 {! `+ R" M8 H
char side; //side表示下子方
* k- k5 ]# a% H* |+ W }; 5 V9 ]5 b1 q0 t. Q
以数组形式保存当前盘面的情况, ) j5 l! i7 K- G1 r8 v2 U
目的是为了在显示当前盘面情况时使用:
2 j# q9 p% m2 B$ J& s) X1 y6 ^ char FiveArea[FIVE_MAX_LINE][FIVE_MAX_LINE];
6 g/ G* }* x* i" `: E b/ t2 s
, ?3 @! f$ c% h6 _: O* T7 f3 O. H 其中FIVE_MAX_LINE表示盘面最大的行数。 6 O5 w$ E: ~% p2 k8 b! s5 a
7 x/ z c. P5 E+ ?' J
同时由于需要在递归搜索的过程中考虑时间和空间有效性,只找出就当前情况来说相对比较好的几个盘面,而不是对所有的可下子的位置都进行搜索,这里用变量CountList来表示当前搜索中可以选择的所有新的盘面情况对象的集合:
# r# Z6 B. X/ u3 E6 f I% b5 c5 a5 E$ U! i
CList CountList; * f J% Q7 L$ j7 o: ~
其中类CBoardSituiton为:
! `" a) I. F! ^" i7 W% L class CBoardSituation 7 a0 r$ o2 n* ]3 s
{ 4 I( T* M/ P* G& Z
CList StepList; //每一步的列表
' h" z2 f, u: z' o3 n& {2 E char FiveArea[FIVE_MAX_LINE][FIVE_MAX_LINE]; 6 J2 z0 k3 t! I1 r( Q9 S: X
struct Step machineStep; //机器所下的那一步
2 N$ C' U+ J$ N! c4 d double value; //该种盘面状态所得到的分数
' u/ w' N7 t5 P3 Z3 X* d, w}
8 t+ l1 l4 h7 W3 i3 I& n/ U1 @8 I* R1 X d+ l
二、评分规则 " R/ L) u0 f+ y) r) E6 Z8 g
对于下子的重要性评分,需要从六个位置来考虑当前棋局的情况,分别为:-,¦,/,\,//,\\
- o* ^1 |) H; G+ B* ^1 P/ y! M9 M& f; R0 k" T" d+ e2 @
" _: _( K: l% o* P7 H& t" A3 ~! p; V 实际上需要考虑在这六个位置上某一方所形成的子的布局的情况,对于在还没有子的地方落子以后的当前局面的评分,主要是为了说明在这个地方下子的重要性程度,设定了一个简单的规则来表示当前棋面对机器方的分数。
0 f& J$ {2 D- q: T, `
& K: H2 l) _ t4 d# u 基本的规则如下:
) W% x! J! v L Q
# O% @. u, X2 f& L; z# ~* n判断是否能成5, 如果是机器方的话给予100000分,如果是人方的话给予-100000 分;
) Q) V, X( O3 Z5 [2 h判断是否能成活4或者是双死4或者是死4活3,如果是机器方的话给予10000分,如果是人方的话给予-10000分;
0 F& z0 W' b* v# R! N# y) Z判断是否已成双活3,如果是机器方的话给予5000分,如果是人方的话给予-5000 分;
/ Y" T$ s' B9 T$ q判断是否成死3活3,如果是机器方的话给予1000分,如果是人方的话给予-1000 分; 7 F' `4 Y$ l+ a. S* |; {
判断是否能成死4,如果是机器方的话给予500分,如果是人方的话给予-500分; + [/ t( S1 V# n* c: n% x0 i9 `
判断是否能成单活3,如果是机器方的话给予200分,如果是人方的话给予-200分;
7 Z9 j( O) q( p" s, [& Y% Z判断是否已成双活2,如果是机器方的话给予100分,如果是人方的话给予-100分; ! p5 s6 `3 \& @
判断是否能成死3,如果是机器方的话给予50分,如果是人方的话给予-50分;
8 ? p" z4 O1 j. D, u判断是否能成双活2,如果是机器方的话给予10分,如果是人方的话给予-10分;
) e! l& h, C; ^; D判断是否能成活2,如果是机器方的话给予5分,如果是人方的话给予-5分;
" L* {/ p" a+ E Y判断是否能成死2,如果是机器方的话给予3分,如果是人方的话给予-3分。
) H M/ _# M0 l7 v% `; ]7 J& n
, e0 {/ I4 y% V( } 实际上对当前的局面按照上面的规则的顺序进行比较,如果满足某一条规则的话,就给该局面打分并保存,然后退出规则的匹配。注意这里的规则是根据一般的下棋规律的一个总结,在实际运行的时候,用户可以添加规则和对评分机制加以修正。
5 w9 B: r3 z; \' p; r: k- A! E/ z- I! P$ E& |2 }9 b
三、胜负判断
: F( R5 X0 g) i8 P 实际上,是根据当前最后一个落子的情况来判断胜负的。实际上需要从四个位置判断,以该子为出发点的水平,竖直和两条分别为 45度角和135度角的线,目的是看在这四个方向是否最后落子的一方构成连续五个的棋子,如果是的话,就表示该盘棋局已经分出胜负。具体见下面的图示:
, ~/ S* S* ?# w Q5 Q' h. Z( n8 N) W5 f0 S, [& D3 y
2 Q' e! ]0 Z( a9 u: f四、搜索算法实现描述
4 Z' X, M+ P) t8 E# Y 注意下面的核心的算法中的变量currentBoardSituation,表示当前机器最新的盘面情况, CountList表示第一层子节点可以选择的较好的盘面的集合。核心的算法如下:
3 O, P# c$ |! W" [8 ~4 R: bvoid MainDealFunction() ) Q/ [" F+ v/ }) N5 g8 a# |
{
7 e1 b. u+ v& I* Q5 ` value=-MAXINT; //对初始根节点的value赋值
1 g$ Y5 S( N& s+ FCalSeveralGoodPlace(currentBoardSituation,CountList);
0 C1 I$ I4 O, _, K9 @//该函数是根据当前的盘面情况来比较得到比较好的可以考虑的几个盘面的情况,可以根据实际的得分情况选取分数比较高的几个盘面,也就是说在第一层节点选择的时候采用贪婪算法,直接找出相对分数比较高的几个形成第一层节点,目的是为了提高搜索速度和防止堆栈溢出。 / ^9 `( |! X6 i, v4 k$ _3 K+ _
pos=CountList.GetHeadPosition();
3 z! w$ {' Q# x7 tCBoardSituation* pBoard;
% _5 ]* x$ u. |2 n- H: j! Efor(i=0;ivalue=Search(pBoard,min,value,0);
; _ \( ]3 V' c- z: Q Value=Select(value,pBoard->value,max);
$ Z! \7 |0 H, C2 V //取value和pBoard->value中大的赋给根节点
( N4 @9 R0 F. w3 P} # M# ^4 b$ P, k7 U, P4 U6 h! Y
for(i=0;ivalue)
. [0 M5 \& {( P, n* T" T) g//找出那一个得到最高分的盘面
( d# z+ C: [) F, D0 u' L { . R( k$ n$ U# g: u$ Q8 k. C
currentBoardSituation=pBoard; 3 r" L t# h# C5 ~% n* R
PlayerMode=min; //当前下子方改为人 0 y! U5 w- k% ?% A" ?& u1 v; e
Break; / P+ |% ]' W5 i* U, s
} % W: [9 e! ]) l3 k, {, t5 @
} `- `2 o, j/ a: n) Y
3 d" R# y1 e0 |- Y! n
其中对于Search函数的表示如下:实际上核心的算法是一个剪枝过程,其中在这个搜索过程中相关的四个参数为:(1)当前棋局情况;(2)当前的下子方,可以是机器(max)或者是人(min);(3)父节点的值oldValue;(4)当前的搜索深度depth。 9 n: W0 \. g8 o
Y+ s% Y7 \3 ]
double Search(CBoardSituation& 7 `1 p: o9 {1 U& g; `
board,int mode,double oldvalue,int depth) , ~& K, {# [8 W- ?3 X$ M
{
* v. v2 j$ T- M; v CList m_DeepList; 2 _* a- j, ~- S
if(deptholdvalue))== TRUE) 1 @" P' b" t. `7 }
{ : Z/ S5 O2 Q2 B# s" L
if(mode==max)
6 L7 m: o- o/ |2 K/ u* w/ l value=select(value,search(successor
+ {- ]3 @, S& ?( ]+ r2 F Board,min,value,depth+1),max);
4 S( x9 S3 |( Q& U$ ~ else , h, k2 L9 ?6 x( V$ ^( W
value=select(value,search(successor
6 X# y/ N/ U" ^; G, W6 m: m: x4 p; s Board,max,value,depth+1),min);
, l% [! C6 W, [* K }
x, `! r H, L. ` return value;
. [1 @8 s2 b: F! I$ y7 J }
# E6 q( d2 S; D& D d" H else & A! `0 f+ Z4 S9 t; o; Z( t6 R
{ ) ^& R* D* f8 v e
if ( goal(board)<>0) - k8 F1 z- o" l" I# k; w, j
//这里goal(board)<>0表示已经可以分出胜负 . p3 U. a: D1 |( E7 ?, C
return goal(board); ( W c! l6 r w# I0 L
else 6 `. v1 C% n q7 ~
return evlation(board);
& D* A; P5 g2 L8 i } . Z( f/ b# n) l3 G* P, c
} % u: C: v, s" i4 i7 x8 G
7 E5 O: M6 V, n: X @! r. d 注意这里的goal(board)函数是用来判断当前盘面是否可以分出胜负,而evlation(board)是对当前的盘面从机器的角度进行打分。
* C: ]1 z- P0 z% R1 Y2 t7 T
: j4 y- D" j$ D6 D3 I 下面是Select函数的介绍,这个函数的主要目的是根据 PlayerMode情况,即是机器还是用户来返回节点的应有的值。 2 v) H& E& l$ }, K9 E, e
% m! U' H% x* Z/ Ddouble Select(double a,double b,int mode)
5 ^) @3 D v& ?. U7 m8 W& ?{
9 c4 k& _( N! N7 Z' r3 w if(a>b && mode==max)¦¦ (a< b && mode==min)
/ Y7 R) U5 F% w* L* ~. f$ H1 k# creturn a;
, h' F* ~' L5 B. O* V else
% S, ?" j1 F: s( Vreturn b; ! h3 h7 v5 [/ T o+ w# o
} 7 E9 b; r' @. f: b. U s
5 I. p# }4 v. g. W8 q五、小结 1 z, [& b$ P5 Z/ j
在Windows操作系统下,用VC++实现了这个人机对战的五子棋程序。和国内许多只是采用规则或者只是采用简单递归而没有剪枝的那些程序相比,在智力上和时间有效性上都要好于这些程序。同时所讨论的方法和设计过程为用户设计其他的游戏(如象棋和围棋等)提供了一个参考。 |
|