本项目是使用java技术+自创“假设下子”算法开发的人机大战四子棋游戏客户端。

 

具体项目,以及原创PSD,设计文档,在文件末尾的百度云连接。

 

 

java三子棋 java四子棋_优先级

 

一. 小组说明:

组名:CST

组长:陈飞良(C):

组员:

沈珂 (S):

谭明航 (T):

 

二.分工说明:

 

①算法思想上:

 

本程序的代码实现思想由三人共同讨论得出,其中组员沈珂的“假设下子”思想尤为精妙,让代码实现更为简单,在这基础上,组员谭明航 ,心思缜密,考虑到各种特殊情况,让整个更加智能。组长陈飞良则负责在他们的基础上设计算法进行完善。

 

②游戏界面上:

 

游戏的整体界面由三人共同讨论,组长陈飞良使用PS制作而成。

 

③代码实现上:

 

Ⅰ.游戏的总体布局与架构由组长陈飞良完成。

 

Ⅱ.本游戏的核心是电脑的下子方法,共有七步

 

①假设该子为电脑子,判断能连成四子(由谭明航实现)

 

②假设该子为玩家子,判断看是否存在,玩家4子相连,即不算此位置玩家将四子的情况,有则拦截(由谭明航实现)

 

③排除垫脚石的状态, 如果此位置上一位置是玩家子,玩家子可成四子,则该位置优先级为-9(由沈珂实现)

 

④排除连子最多,但是到顶端也无法组成4个连子的(由谭明航实现)

 

⑤如果电脑放一子下一步 电脑能同时形成2种方法成四子 ,则设置max_x[x] = 4,即优先级为 4。(由陈飞良实现)

 

⑥如果玩家放一子下一步玩家能同时形成2种方法成四子则拦截 ,则设置max_x[x] = 10,即优先级为 10(由陈飞良实现)

 

⑦普通攻击,假设电脑按照优先级先后(在max_x中有各列优先级),同级则随机落子。(由沈珂实现)

 

三.课程设计思路说明

I.各个类

本程序共设计了一个6个类,主要说下3个类

1.

其中有位置类,即Position类。并且Position类中有一个整型标志成员Label_Status.

 

Label_Status = 0 表示空 .

 

Label_Status = 1 表示玩家子.

 

Label_Status = 2 表示电脑子.

2.

Game类为主类,放了布局的各个控件,以及,玩家和电脑的移动方法等。

3.

MainPanel类为主面板

II.各个操作的实现

 

1.初始化

 

在设计主界面初始化时先将每个格子放下代表玩家子和电脑子的Label成员.同时设置为不可见,并且Label_Status = 0 . 即该位置为空

 

2.下子

 

实现方法:使相应位置的玩家Label或者电脑Label可见,同时更改标志。

 

玩家的下子:通过监听键盘和鼠标进行相应的移动,获得玩家下子的位置

 

III.电脑的下子

 

电脑的下子是整个程序的核心,基本是遵循人的思维

 

设置一个_y[]数组每个元素初始化为-1,表示该列无可下位置

 

1.找到每列可以下的位置

 

遍历某一列的每一行( x : 0-- 6),找到能下子的那一行,即该子的y坐标

 

找到后位置赋值给_y[],即 _y[x] = y’;

 

当然此处的y坐标对应的位置应该为115 + 100 * y

 

同时x坐标对应的位置为375 + 100 * x

 

(这个位置由棋盘大小,和布局时埋子的方式决定)

 

2.找出各列放电脑子后可成最大连子数

 

定义了一个max_x存储每一列(x : 0 --- 7)所下位置各方向可成最大连子,每个元素初始化为-10(只是取一个小于0的数可以随意取,但是要方便后期的优先级设置,可以为-7 -8 等)

 

使用三次Math.max可求出

 

max_x[r] = Math.max(Math.max(you, you_xia), Math.max(xia, you_shang));

 

3.考虑各种特殊情况

 

①假设该子为电脑子,判断能连成四子

 

this.pos[x][y].setLabel_Status(2);

 

遍历查找,成立,则落子,直接宣布比赛结果。

 

如果不可以,标志恢复

 

this.pos[x][y].setLabel_Status(0);

 

以下各种情况同理

 

②假设该子为玩家子,判断看是否存在,玩家4子相连,即不算此位置玩家将四子的情况,有则拦截

 

③排除垫脚石的状态, 如果此位置上一位置是玩家子,玩家子可成四子,则该位置优先级为-9

 

把该位置的max_x[x] = -9;  (前期max_x元素初始化为-10)即除非没地方下,才下此处

 

④排除连子最多,但是到顶端也无法组成4个连子的

 

max_x[x] = 0; 优先级比一般的小,但比垫脚石高

 

⑤如果电脑放一子下一步 电脑能同时形成2种方法成四子 ,则设置max_x[x] = 10,即优先级为 4

 

⑥如果玩家放一子下一步玩家能同时形成2种方法成四子则拦截 ,则设置max_x[x] = 10,即优先级为 10

 

⑦普通攻击,假设电脑下子后,连子最多的位置(在max_x中有各列最大)随机落子。(前面的垫脚石类的情况,可以改变max_x中的值来改变优先级)

 

如果是2--4列   max_x += 0.5 ;   同等优先级,中间再优先一点,且不会跨级。

 

可创建一个数组a,把max_x数据赋值过去,然后利用Arrays.Sort方法排序最后去a最后一个元素,即为a的最大值,即max_x最大值

 

具体代码编写顺序如下:

 

 

java三子棋 java四子棋_优先级_02

 

 

 

 

四.程序的成长过程,特点,以及需要改进的地方

 

1.成长过程

 

①.自我完善

 

由于之前照着网上的例子敲过人与人对战的五子棋,所以完成的效率非常的高,在老师说公布题目之后,我们组便开了的个会,把基本的算法定下了。组员沈珂的“假设下子”思想,对我们的代码实现作用非常之大,在这第一次会议中便确定了7个主要的方法,同时在一周后,便写出了1.0版。然后一路的查漏补缺。

java三子棋 java四子棋_代码实现_03

 

 

java三子棋 java四子棋_初始化_04

 

java三子棋 java四子棋_代码实现_05

 

 

 

java三子棋 java四子棋_java三子棋_06

 

②.取长补短

 

本程序曾与两个网站的游戏进行对垒

 

(1)一个是 http://www.4399.com/flash/48470_1.htm ,这个网站算法有个优点就是懂得占据中间位置,比过之后,便决定在我们2--4列的优先级+0.5,使其在能形成同样多连子数的时候,比其他位置优先,尽量占据中间位置,但是其他位置比下此位置连子多时,仍然会下那个连子更多的位置。由于这个网站没有写垫脚石的算法,所以我们改完之后的胜率是80%左右。

 

(2)第二个是   http://www.7k7k.com/swf/50447.htm ,这个网站的优点是能主动形成双三,比过之后,我们便在我们程序中加了两个方法:

 

 

java三子棋 java四子棋_java三子棋_07

 

 

但是,这个网站的方法还有一个方法便是,在下一个子之后成三子,我们被迫去拦截,同时还给他自己垫脚,此时我们必输。

 

同时,我们程序没有写防止自己给对方垫脚,使其成双三的算法:

 

 

 

所以,在完善后,我们的胜率只能在50--60%之间徘徊。

 

2.程序的特点

 

本程序的”假设下子”思想是采用枚举的方法优点在于不需要考虑具体情况,如:

 

 

java三子棋 java四子棋_java三子棋_08

 

 

在方法6中,可以拦截的情况有:

 

 

 

1.水平将有两种方法成4子

 

 

 

 

java三子棋 java四子棋_java三子棋_09

 

 

 

 

 

 

2.斜方向上将有两种方法成4子

 

java三子棋 java四子棋_初始化_10

 

 

 

 

 

 

 

 

 

3.双方向成3子

 

 

java三子棋 java四子棋_java三子棋_11

 

 

 

 

...........等等。即只要符合下一子,能有两种方法成四子就拦,并不需要考虑具体情况,由电脑“假设下子”去枚举,代码效率非常的高,与其他组的程序对比,基本以可以20行代码发挥200行代码的作用。

 

3.需要改进的地方

(1)完善方法8

(2)写出,提前拦截 玩家在下一个子之后成三子,导致我们被迫去拦截,同时还给他自己垫脚 的方法

 

项目连接  https://pan.baidu.com/s/1mM95FgUyeHWT6AhY8hA8CA