实践出真知,为了学习JAVA编程,笔者购买了《疯狂JAVA实战演义》一书,打算从这本书开始学习,选择它的原因是书中介绍了15个Java项目,主要是一些游戏和工具。兴趣是最好的老师,这些项目很吸引我,学习过程也快乐了许多。

于是打算把我读书和实践的过程记录下来,也算是一种积累,一种纪念。

从遇到的第一个项目说起,“控制台五子棋”。该项目的目的是帮助读者掌握和理解Java编程的一些基本知识,为了能更好的突出这一学习目的,这个项目没有实现漂亮的界面和方便易用的人机交互功能,也没有过多地考虑程序的扩展以及设计模式的使用。

作者在总结一节中写得很清楚:读者朋友,从这个项目里你可以学到数组、枚举类的使用方法,获取键盘输入的方法,分支流程、循环流程,当然还有获得满足感和快乐感。

我问自己,我想从这个项目中得到什么?

我想要解决一个问题,那就是:我要编程,我该从哪里开始,如何下手?

第一步:理解五子棋的游戏规则

中国象棋、国际象棋、围棋也好,五子棋也好,在编程之前,要知道它们的游戏规则,五子棋我玩过,我将其游戏规则描述为“把自己的棋子连成五个就算赢了”。我看完了书作者的描述,一方面我有一种坐享其成的感觉,一方面我觉得自己的描述实在是太草率了。“把自己的棋子连成五个就算赢了”这种描述对编程实在是没有什么太大的作用!

现在让我再重新思考一下五子棋的游戏规则:

玩五子棋之前的准备工作:两个人,一个棋盘,两盒棋子(可能一盒是黑色的,另一盒是白色的)。

棋盘有多大,棋子有多少呢?

两个人必须都很清楚游戏规则,也就是棋子下在哪里,谁先下,何种情况下算赢?

首先在横或竖或斜的方向上形成五个子相连的一方为赢。

第二步:从现实世界到计算机世界,考虑程序输入与输出。

为了简单起见,两人对弈改为人与“电脑”对弈。棋盘用一个二维字符串数组存储,玩家以(x,y)的方式从键盘输入棋子落在棋盘上的坐标,棋盘和棋子打印在控制台,黑棋子用形如黑色圆点的特殊字符表示,白棋子用形如白色圆点的特殊字符表示,棋盘的格子用十字表示,真是太有才了!棋子落在棋盘上就是用表示棋子的字符代替十字。

第三步:绘制游戏流程

所谓一图胜千言,注意图中这几类点:开始点、结束点、判定点。

从这张图中,可以看出几个我们在编程过程中要解决的明显的问题:

1.如何从键盘输入x,y坐标?

2.如何判断玩家或是“电脑”赢了?

3.如何实现“电脑”自动下棋?

4.胜出后,如何开始一轮新的游戏?

5.胜出后,如何退出游戏?

还有一些不明显的问题:

1.如何验证玩家的输入是否合法有效?

2.如何实现玩家与“电脑”的轮流下棋?

第四步:用面向对象的思想设计类

棋盘类(Chessboard)、棋子类(Chessman)两个类很容易想出来,但是光有这两个类还不够,有些操作这两个类处理不了,比如玩家输入x,y坐标,放在棋盘类和棋子类中都不合适,于是我有一次坐享其成了,作者搞了个游戏类(GobangGame)出来,完成一系列的操作,比如游戏的开始与结束、玩家坐标输入及验证、判断游戏是否胜出、电脑下棋。游戏类(GobangGame)是最主要的一个类,它的方法最多,能改进的地方也最多,下面展示了五子棋类图。

第五步:实现类,搭积木

棋盘类(Chessboard)的核心功能就是用来维护棋盘,在“第二步:从现实世界到计算机世界,考虑程序输入与输出”中已经提到,棋盘在计算机世界里是用一个二维字符串数组存储的,所以维护棋盘,就是创建、初始化、修改、获取这个二维字符串数组的过程,为什么选用数组,至少有两点,其一,棋盘的大小是固定的,没有人会去偷走棋盘上的格子对吗?其二,二维数组的下标正好表示了棋盘的位置(x,y),数组元素的值正好可以存储棋盘的状态:黑子、白子、无子。其三,可以通过数组的下标很容易的找到数组元素。

棋子类(Chessman)真是让我着了下急,作者是用枚举类实现的,枚举类我不熟啊,很少用,但这种困难是对知识点不熟造成的,不能算作难点。作者考虑用枚举类来实现的原因思考了下,因为棋子在计算机中使用固定的两个字符串对象表示的,如何每次下一个棋子,都要重新创建一个字符串对象,是不是太浪费了,黑子和黑子在计算机里没有差别的。

之前两个类讲到了两个在Java编程中很基本也很重要的知识点:数组和字符串。下面看一看游戏类(GobangGame)。

对于游戏类(GobangGame)的实现感觉有点面向过程,关键是要理顺游戏的流程。

使用while语句、continue、break来达到循环接收玩家输入、跳出本次循环、跳出整个循环。使用标志变量isOver来标示本次游戏是否胜出。

游戏类(GobangGame)在实现过程中值得注意的有:

1.利用BufferedReader、InputStreamReader、System.in、readLine()获取键盘的输入;

2.凡是涉及玩家输入的部分,一定要对输入的内容进行验证;

3.判断玩家或“电脑”是否赢了,需要一个胜出算法;

4.如果游戏胜出,提示哪一方赢了,并提示玩家是否开始下一轮游戏或退出整个游程序;

5.在合适的时机重新初始化棋盘。

对于获取键盘的输入还有对输入的内容进行验证是固定的套路,没有什么可多说的。

关键是胜出算法,可以说在这里算是一个难点,需要动一动脑筋。我一直以来对算法都是一种逃避的态度,总觉得那不是一般人能想出来的。这一次我也是皱起了眉头。

如果让我下棋,我可以判断是否胜出,但是让计算机也能判断胜出,我就不会了。

我尽力去回忆自己的大脑是如何判断的,遍历每一个方向并计数,每次把当前棋子作为五子中的一员,使它的位置不停地发生变换,这就是我的大脑判断的过程。

开始计数后,若遇到相同的棋子,计数器就加1;若遇到不同的棋子,就将计数器归零。每次计数后都检查计数器是否为五,若计数器计数值为五,说明找到了连续的五个相同的棋子(五连子),直接返回即可。

public boolean isWon(String chessman){int startX=0;int startY=0;int endX=ChessBoard.Board_Size-1;int endY=ChessBoard.Board_Size-1;//     最小横坐标中间值int minTmpX=posX-WIN_COUNT+1;//       最小纵坐标中间值int minTmpY=posY-WIN_COUNT+1;//       最大横坐标中间值int maxTmpX=posX+WIN_COUNT-1;//       最大纵坐标中间值int maxTmpY=posY+WIN_COUNT-1;//直线上最小横坐标int minX=minTmpX>0?minTmpX:0;//直线上最小纵坐标int minY=minTmpY>0?minTmpY:0;//直线上最大横坐标int maxX=maxTmpX<endX?maxTmpX:endX;//直线上最小纵坐标int maxY=maxTmpY<endY?maxTmpY:endY;//1.在横线方向上检查是否胜出,胜出返回trueint sameCount=0;for(int i=minY;i<=maxY;i++){String[][]brd = chessboard.getboard();//若棋子不同则计数归零if(!(brd[posX][i].equals(chessman)))sameCount=0;//若棋子相同则累加else sameCount++;if(sameCount==WIN_COUNT)return true;}//2.在纵线方向上检查是否胜出,胜出返回true,与横线方向上相似for(int i=minX;i<=maxX;i++){String[][]brd = chessboard.getboard();//若棋子不同则计数归零if(!(brd[i][posY].equals(chessman)))sameCount=0;//若棋子相同则累加else sameCount++;if(sameCount==WIN_COUNT)return true;}//3.在左上右下斜线方向上检查是否胜出,胜出返回truefor(int i=minX,j=minY;(i<=maxX&&j<=maxY);i++,j++){String[][]brd = chessboard.getboard();//若棋子不同则计数归零if(!(brd[i][j].equals(chessman)))sameCount=0;//若棋子相同则累加else sameCount++;if(sameCount==WIN_COUNT)return true;}//4.在右下左上斜线方向上检查是否胜出,胜出返回truefor(int i=maxX,j=minY;(i>=minX&&j<=maxY);i--,j++){String[][]brd = chessboard.getboard();//若棋子不同则计数归零if(!(brd[i][j].equals(chessman)))sameCount=0;//若棋子相同则累加else sameCount++;if(sameCount==WIN_COUNT)return true;}return false;}

贴出这段算法的代码,请读者检验,如果您有更好的算法,或者我的算法哪里写的有误,也欢迎学习交流。

该算法主要有三个关键点,第一点是求最大横坐标、最小横坐标、最大纵坐标、最小纵坐标,第二点是求在横向、纵向、左上右下、右上左下是否有五连子,第三点是在斜向上,当遇到棋盘的四条边时(除四个角上的位置),如何处理遍历的开始点和结束点。

虽然简单实现了控制台五子棋的功能,但作为一个游戏来讲,它有很多局限性,甚至还有错误,笔者尝试着提供和实现一些解决办法。

局限性一:

这里玩家的对手“电脑”下棋是随机的,它每次下棋的位置是没有一个算法来支撑的,而是靠随机函数Math.random(),所以游戏的的可玩性很差,玩家很容易就取胜。

局限性二:

在“电脑”下棋时,每随机生成一个棋子的(x,y)坐标,都要检查该位置是否已经被其它棋子占用,如果占用需要重新生成一个棋子的(x,y)坐标,程序中使用了while循环,如果棋盘上所有的位置都被占满了,程序就会陷入死循环。

解决办法:

定义一个数组保存每个位置的状态,定义一个计数器保存被占用的位置的个数,每次都检验计数器的数字是否达到上限,如果达到最大个数,说明棋盘已被占满,可提示玩家和棋,然后重新开始游戏。

局限性三:

在控制台中,使用字符串表示棋子,如果使用Swing来开发图形界面的话,可以建立棋子接口,提供黑棋和白棋的实现类,棋盘二维数组中存放的不在是字符串对象,而是接口,如果有新种类的棋子时,就不需要修改棋盘类了。

在编写程序的过程中还发现了自己对一个知识点的混淆:

在“电脑”生成新坐标时,使用了这样一段代码:

while(board[pos[0]][pos[1]]!="十"){pos[0] = (int)(Math.random()*(ChessBoard.Board_Size));pos[1] = (int)(Math.random()*(ChessBoard.Board_Size));
} 

再看下board的定义:

public void initBoard(){for(int i=0;i<Board_Size;i++){for(int j=0;j<Board_Size;j++)board[i][j] = new String("十");}
}

这个while语句总是为真,程序进入了死循环。

== 是关系运算符,用于判断两个简单变量的值是否相等,或两个引用变量的引用地址是否相等。

board[pos[0]][pos[1]]的值是引用地址,所以它总是不等于“十”。

equals()是用于判断引用变量引用地址指向的存储内容是否相等。

修改后的代码:

while(!board[pos[0]][pos[1]].equals("十")){pos[0] = (int)(Math.random()*(ChessBoard.Board_Size));pos[1] = (int)(Math.random()*(ChessBoard.Board_Size));}  

控制台五子棋--学习笔记相关推荐

  1. C语言五子棋人人对弈学习笔记

    C语言编写五子棋人人对弈学习笔记 1.头文件#include <conio.h> #include <conio.h>是一个控制输出的头文件. 包含以下函数:textbackg ...

  2. Java学习笔记(十)--控制台输入输出

    输入输出 一.控制台输入 在程序运行中要获取用户的输入数据来控制程序,我们要使用到 java.util 包中的 Scanner 类.当然 Java 中还可以使用其他的输入方式,但这里主要讲解 Scan ...

  3. Qt 学习笔记(5)绘图 五子棋游戏

    在上一篇博客C++ Qt学习笔记(4)绘图中介绍了Qt中的绘图方法,基于上一篇的博客的知识,使用QPainter设计一个五子棋的棋盘,后续会完成五子棋的游戏设计. 1. 棋盘的设计 首先需要绘制棋盘的 ...

  4. 五子棋人机大战(Java菜鸟学习笔记)

    五子棋人机大战(Java菜鸟学习笔记) 逻辑是: 1.绘制棋盘: 2.人机开始各执行一次操作,每进行一次操作就进行判断输赢 其中判断输赢主要分成四个方向,横向.纵向以及交叉向,此时可以画图来找规律实现 ...

  5. DJL初学者学习笔记(一):Java启动DJL控制台打印No matching cuda flavor for win found: cu65并且在线下载dll文件解决方案

    1.举例,来自知乎深度学习库的事例https://zhuanlan.zhihu.com/p/396524552 2.修改,下载之后做略微的修改指定model文件 Criteria<NDArray ...

  6. 【JavaScript学习笔记2】JS中常见的输出方式-控制台输出信息

    引言 在编程开发的过程中,输出信息是非常必要的.JS中提供了四种输出方式:弹出显示框.控制台输出.弹出输入框.弹出判断显示框 弹出显示框 这种方式在上一篇笔记中已经详细介绍,有需要学习的朋友可以跳转到 ...

  7. 容器云原生DevOps学习笔记——第三期:从零搭建CI/CD系统标准化交付流程

    暑期实习期间,所在的技术中台-效能研发团队规划设计并结合公司开源协同实现符合DevOps理念的研发工具平台,实现研发过程自动化.标准化: 实习期间对DevOps的理解一直懵懵懂懂,最近观看了阿里专家带 ...

  8. NuGet学习笔记(3) 搭建属于自己的NuGet服务器

    文章导读 创建NuGetServer Web站点 发布站点到IIS 添加本地站点到包包数据源 在上一篇NuGet学习笔记(2) 使用图形化界面打包自己的类库 中讲解了如何打包自己的类库,接下来进行最重 ...

  9. kvm虚拟化学习笔记(十七)之KVM到KVM之v2v迁移

    1.源KVM虚拟主机node1 (1).查看源KVM虚拟主机上的虚拟机列表,本文计划将CentOS6.5-01虚拟机迁移到其它KVM虚拟主机中. [root@node1 ~]# virsh list ...

最新文章

  1. Spring Boot 2.X 对 web 的开发支持(二)
  2. ie8 js未指明的错误_修复ueditor百度编辑器在IE8下shCore.js报错'undefined'错误的问题...
  3. 联想确认再次裁员 称调整主要分布在海外
  4. 哈达玛变换的应用SATD、SAD等匹配算法
  5. BAE3.0还不支持本地写入文件
  6. 多个vue项目合并成一个_再见Vlookup,合并多个表格发现一个最简单方法
  7. 数据结构 10分钟让你掌握经典排序(一)
  8. Android TextView 显示HTML加图片
  9. 购物车单选全选,计算总价,出现个小问题,没找到.....
  10. wincc服务器客户端用虚拟机,什么情况下用wincc服务器与客户端
  11. ZMQ中线程之间发送命令
  12. 查找Ubuntu下包的归属
  13. c语言例题22:日期计算
  14. Teststand 中用labview 读写station options属性
  15. SQL窗口函数OVER用法整理
  16. debian7升级到debian9
  17. 掌握Android图像显示原理(上)
  18. 复杂网络的学习——抗毁性
  19. 2017年美国50家最顶尖的初创公司排行榜
  20. Codeforces Daily (Round 370-410)

热门文章

  1. 2022-2028全球与中国3D智能手机市场现状及未来发展趋势
  2. 外贸独立站该如何选择跨境支付方式?
  3. jmeter的json提取器和json提取器取值
  4. 计算机行业个人就业意向,个人应聘工作简历-计算机专业.docx
  5. 在windows server 2012上搭建DNS服务器
  6. 小米2 android版本更新,小米2手机怎么升级MIUI系统?小米2在线升级的教程
  7. Android8.1.0安卓源码编译
  8. SVN上箭头都是什么意思,红色灰色蓝色的
  9. gitlab搭建问题
  10. vue+elementui中使用echarts给柱形图添加背景色