package com.test;

 public class Chessboard {
     //定义二维数组作为棋盘
     private String[][] board;
     //定义棋盘大小
     public static final int BOARD_SIZE=22;

     //初始化棋盘
     public void initBoard(){
         board=new String[BOARD_SIZE][BOARD_SIZE];//创建二维数组
         //定义初值,嵌套循环
         for(int i=0;i<BOARD_SIZE;i++){
             for(int j=0;j<BOARD_SIZE;j++){
                 board[i][j]="十";
             }
         }
     }
     //test函数的意义不明
     public void test(){
         Object[][] array=new Object[10][10];
         for(int i=0;i<array.length;i++){
             for(int j=0;j<array.length;j++){
                 array[i][j]=new Object();
             }
         }
     }
     //打印棋盘
     public void printBoard(){
         for(int i=0;i<BOARD_SIZE;i++){
             for(int j=0;j<BOARD_SIZE;j++){
                 System.out.print(board[i][j]);
             }
             System.out.print("\n");
         }
     }
     //放置棋子
     public void setBoard(int posX,int posY,String chessman){
         this.board[posX][posY]=chessman;
     }
     //返回棋盘
     public String [][] getBoard(){
         return this.board;
     }

 }

Chessboard.java

 package com.test;

 public enum Chessman {
     BLACK("●"),WHITE("○");
     //设置两个静态棋子属性
     //等同于public static final Chessman BLACK=new Chessman("●")
     private String chessman;
     //设置后可通过.BLACK.chessman进行设置棋子种类
     private Chessman(String chessman){
         this.chessman=chessman;
     }
     //返回棋子
     public String getChessman(){
         return this.chessman;
     }
 }

Chessman.java

以上为棋盘类(Chessboard),棋子类(Chessman,枚举类)

最后是游戏类GobangGame.java

 package com.test;

 import java.io.BufferedReader;//输入
 import java.io.InputStreamReader;//输出

 public class GobangGame {
     private final int WIN_COUNT=5;//设置获胜棋子数
     private int posX=0;
     private int posY=0;
     private Chessboard chessboard;//Chessboard类的一个实例

     //将chessboard实例调用
     public GobangGame(Chessboard chessboard){
             this.chessboard=chessboard;
     }
     //接收inputStr里面的x,y值,并赋给posX/Y
     public boolean isValid(String inputStr){
         String[] posStrArr=inputStr.split(",");
         //将字符串分割成字符数组,分割标志为“,”
         try{
             posX=Integer.parseInt(posStrArr[0])-1;
             posY=Integer.parseInt(posStrArr[1])-1;
         }catch(NumberFormatException e){
             chessboard.printBoard();
             System.out.println("pls input position like num,num");
             return false;
         }
         if(posX<0||posX>=Chessboard.BOARD_SIZE||posY<0||posY>=Chessboard.BOARD_SIZE){
             chessboard.printBoard();
             System.out.println("reinput");
             return false;
         }
         //接收chessboard
         String[][] board=chessboard.getBoard();

         if(board[posX][posY]!="十"){
             chessboard.printBoard();
             System.out.println("re input");
             return false;
         }
         return true;
     }
     //游戏主流程函数,配图
     public void start()throws Exception{
         boolean isOver=false;
         chessboard.initBoard();
         chessboard.printBoard();
         BufferedReader br=new BufferedReader(new InputStreamReader(System.in));
         String inputStr=null;
         //若输入不为空,开始循环
         while((inputStr=br.readLine())!=null){
             isOver=false;
             //验证有效(是棋盘坐标)
             if(!isValid(inputStr)){
                 continue;
             }
             //设定棋子
             String chessman=Chessman.BLACK.getChessman();
             chessboard.setBoard(posX, posY, chessman);
             //判定是否胜利
             if(isWon(posX,posY,chessman)){
                 isOver=true;
             }else{
                 int[] computerPosArr=computerDo();
                 chessman=Chessman.WHITE.getChessman();
                 chessboard.setBoard(computerPosArr[0], computerPosArr[1], chessman);
                 if(isWon(computerPosArr[0],computerPosArr[1],chessman)){
                     isOver=true;
                 }
             }
             //判定是否结束
             if(isOver){
                 if(isReplay(chessman)){
                     chessboard.initBoard();
                     chessboard.printBoard();
                     continue;
                 }
                 break;
             }
             //进行下一回合
             chessboard.printBoard();
             System.out.println("pls input the num");
         }
     }
     //是否重新开始游戏
     public boolean isReplay(String chessman)throws Exception{
         chessboard.printBoard();
         String message=chessman.equals(Chessman.BLACK.getChessman())? "you win":"you lose";
         System.out.println(message+"paly again?(Y/N)");
         BufferedReader br=new BufferedReader(new InputStreamReader(System.in));
         if(br.readLine().equals("y")){
             return true;
         }
         return false;
     }
     //生成电脑棋子(随机生成即可,无逻辑)
     public int[] computerDo(){
         int posX=(int)(Math.random()*(Chessboard.BOARD_SIZE-1));
         int posY=(int)(Math.random()*Chessboard.BOARD_SIZE-1);
         String[][] board=chessboard.getBoard();
         //验证电脑下一步棋的坐标没有棋子
         while(board[posX][posY]!="十"){
             posX=(int)(Math.random()*Chessboard.BOARD_SIZE-1);
             posY=(int)(Math.random()*Chessboard.BOARD_SIZE-1);
         }
         int[] result={posX,posY};
         return result;
     }
     //胜利判定(方法为在当前所下棋子周围进行检查,非遍历棋盘法)
     public boolean isWon(int posX,int posY,String ico){
         int startX=0;//直线开始X坐标
         int startY=0;//直线开始Y坐标
         int endX=Chessboard.BOARD_SIZE-1;//直线结束X坐标
         int endY=endX;//直线结束Y坐标
         int sameCount=0;//同条直线上相邻棋子数
         int temp=0;
         //计算棋子的最小X坐标和Y坐标
         temp=posX-WIN_COUNT+1;
         startX=temp<0?0:temp;//若小于0,则为0
         temp=posY-WIN_COUNT+1;
         startY=temp<0?0:temp;
         //计算棋子的最大X坐标和Y坐标
         temp=posX+WIN_COUNT-1;
         endX=temp>Chessboard.BOARD_SIZE-1?Chessboard.BOARD_SIZE-1:temp;
         temp=posY+WIN_COUNT-1;
         endY=temp>Chessboard.BOARD_SIZE-1?Chessboard.BOARD_SIZE-1:temp;
         //从左到右方向计算相同相邻棋子的数目
         String[][] board=chessboard.getBoard();
         //竖直方向
         for(int i=startY;i<endY;i++){
             if(board[posX][i]==ico&&board[posX][i+1]==ico){
                 sameCount++;
             }else if(sameCount!=WIN_COUNT-1){
                 sameCount=0;
             }
         }
         //水平方向
         if(sameCount==0){
             for(int i=startX;i<endX;i++){
                 if(board[i][posY]==ico&&board[i+1][posY]==ico){
                     sameCount++;
                 }else if(sameCount!=WIN_COUNT-1){
                     sameCount=0;
                 }
             }
         }
         if(sameCount==0){
             int j=startY;
             for(int i=startX;i<endX;i++){
                     if(j<endY){
                         if(board[i][j]==ico&&board[i+1][j+1]==ico){
                             sameCount++;
                         }else if(sameCount!=WIN_COUNT-1){
                             sameCount=0;
                         }
                         j++;
                     }
                 }
             }
             return sameCount==WIN_COUNT-1?true:false;
         }

     public static void main(String[] args)throws Exception{
         GobangGame gb=new GobangGame(new Chessboard());//创建实例
         gb.start();//开始程序
     }
 }
     

其中的start()函数写了多个if语句,可读性较差。附上流程图(^…^!手写渣)

-END

-^^^^^^^^^^^^^^^^^^

-人世几回伤往事,山形依旧枕寒流。

-2016-09-06

疯狂Java讲义 第一章控制台五子棋(代码分析)的更多相关文章

  1. 《疯狂java讲义》笔记 1-5章

    1.编译语言和解释语言理解,摘自李刚老师的<疯狂Java讲义>第三版: 就是说,Java和.net都是编译型有事解释型语言.编译型就是根据不同平台编译成不同的可执行机器码,编译过程中会进行 ...

  2. 《疯狂Java讲义第4版》PDF+代码+课件 电子书pdf 分享

    <疯狂Java讲义(第4版)>是<疯狂Java讲义>的第4版,第4版保持了前3版系统.全面.讲解浅显.细致的特性,全面新增介绍了Java 9的新特性. <疯狂Java讲义 ...

  3. 0038 Java学习笔记-多线程-传统线程间通信、Condition、阻塞队列、《疯狂Java讲义 第三版》进程间通信示例代码存在的一个问题

    调用同步锁的wait().notify().notifyAll()进行线程通信 看这个经典的存取款问题,要求两个线程存款,两个线程取款,账户里有余额的时候只能取款,没余额的时候只能存款,存取款金额相同 ...

  4. 疯狂JAVA讲义---第十二章:Swing编程(五)进度条和滑动条

    http://blog.csdn.net/terryzero/article/details/3797782 疯狂JAVA讲义---第十二章:Swing编程(五)进度条和滑动条 标签: swing编程 ...

  5. 疯狂JAVA讲义第三章之数组篇

    package test;   /** * Desription: * @author orangebook *<br/>网站:<a href="http://www.cr ...

  6. udp协议基础(转自疯狂java讲义)

    第17章  网络编程 17.4  基于UDP协议的网络编程 UDP协议是一种不可靠的网络协议,它在通信实例的两端各建立一个Socket,但这两个Socket之间并没有虚拟链路,这两个Socket只是发 ...

  7. 《疯狂java讲义》读后感

    <疯狂java讲义·第三版>,全书共851页,18章. 目录如下: 第1章 Java语言概述与开发环境 第2章 理解面向对象 第3章 数据类型和运算符 第4章 流程控制与数组 第5章 面向 ...

  8. 【Java】-NO.16.EBook.4.Java.1.007-【疯狂Java讲义第3版 李刚】- Java基础类

    1.0.0 Summary Tittle:[Java]-NO.16.EBook.4.Java.1.007-[疯狂Java讲义第3版 李刚]-  Java基础类 Style:EBook Series:J ...

  9. 【Java】-NO.16.EBook.4.Java.1.008-【疯狂Java讲义第3版 李刚】- 集合/容器

    1.0.0 Summary Tittle:[Java]-NO.16.EBook.4.Java.1.008-[疯狂Java讲义第3版 李刚]- 集合 Style:EBook Series:Java Si ...

随机推荐

  1. 用户 'IIS APPPOOL\DefaultAppPool'登录失败

    今天发布网站遇到这个问题.问题直接说明iis  应用程序池. 后来百度发现是应用程序池  进程模型中的标识项设置问题,这个我用的是本地数据库所以是localsystem.在此小弟谢谢这位 http:/ ...

  2. 实战Lucene,初始Lucene

    实战 Lucene,第 1 部分: 初识 Lucene 本文首先介绍了 Lucene 的一些基本概念,然后开发了一个应用程序演示了利用 Lucene 建立索引并在该索引上进行搜索的过程. 10 评论: ...

  3. win8下 web测试 之 hosts绑定

    从这个开始,开启web测试之旅 绑定hosts: 1.在C:\Windows\System32\drivers\etc下找到 hosts 文件 2.将hosts文件复制到一个地方: 3.修改hosts ...

  4. Repository,UnitOfWork,DbContext(1)

    一.前言 终于到EF了,实在不好意思,最近有点忙,本篇离上一篇发布已经一个多星期了,工作中的小迭代告一段落,终于有点时间来继续我们的架构设计了,在这里先对大家表示歉意. 其实这段时间我并不是把这个系列 ...

  5. JDBC_mysql---防sql注入,存储图片

    package PreparedStatement_sql注入; import java.io.File; import java.io.FileInputStream; import java.io ...

  6. SQLServer服务器数据库之间的数据操作(完整版)

    分类: 数据库开发技术 ---------------------------------------------------------------------------------- -- Au ...

  7. C语言中宏定义(#define)时do{}while(0)的价值(转)

    C语言中宏定义(#define)时do{}while(0)的价值 最近在新公司的代码中发现到处用到do{...}while(0),google了一下,发现Stack Overflow上早有很多讨论,总 ...

  8. eclipse使用Git插件

    折腾了会Git,记录一下下. 1.安装Git  Help-->Install New Software  点击Add,Name随意,Location为http://download.eclips ...

  9. C++_基础_C与C++的区别

    内容: (1)C++简介和编程的基本变化 (2)命名空间的概念和使用 (3)结构体.联合.枚举的不同 (4)布尔类型 以及 运算符别名 (5)函数的重载.缺省参数.哑元以 及内联 1.简介和编程的基本 ...

  10. 多线程程序中fork导致的一些问题

    最近项目中,在使用多线程和多进程时,遇到了些问题. 问题描述:在多线程程序中fork出一个新进程,发现新的进程无法正常工作. 解决办法:将开线程的代码放在fork以后.也就是放在新的子进程中进行创建. ...