iis服务器助手广告广告
返回顶部
首页 > 资讯 > 精选 >Java怎么实现简单的五子棋游戏
  • 378
分享到

Java怎么实现简单的五子棋游戏

2023-06-30 13:06:29 378人浏览 泡泡鱼
摘要

本文小编为大家详细介绍“Java怎么实现简单的五子棋游戏”,内容详细,步骤清晰,细节处理妥当,希望这篇“Java怎么实现简单的五子棋游戏”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。项目结构这个是在网上找的资源,

本文小编为大家详细介绍“Java怎么实现简单的五子棋游戏”,内容详细,步骤清晰,细节处理妥当,希望这篇“Java怎么实现简单的五子棋游戏”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。

项目结构

这个是在网上找的资源,出处记不得了,记录一下。程序的总体结构,很简单的:

Java怎么实现简单的五子棋游戏

核心代码

代码如下:

ArrComparator.java类

import java.util.Comparator;class ArrComparator implements Comparator<Object> {    int column = 2;    int sortOrder = -1; // 递减    public ArrComparator() {    }    public int compare(Object a, Object b) {        if (a instanceof int[]) {            return sortOrder * (((int[]) a)[column] - ((int[]) b)[column]);        }        throw new IllegalArgumentException("param a,b must int[].");    }}

CheSSMap.java类

import javax.swing.*;import java.awt.*;import java.awt.event.*;import java.net.URL;@SuppressWarnings("serial")public class ChessMap extends JFrame {private ImageIcon map;//棋盘背景位图private ImageIcon blackchess;//黑子位图private ImageIcon whitechess;//白子位图private ChessPanel cp;//棋盘private JPanel east;private JPanel west;private static final int FINAL_WIDTH = 450;private static final int FINAL_HEIGHT = 500;//以下为下拉菜单private JMenuBar menubar;private JMenu[] menu={new JMenu("开始"),new JMenu("设置"),new JMenu("帮助")};private JMenuItem[] menuitem1={new JMenuItem("重新开始"),new JMenuItem("悔棋"),new JMenuItem("退出")};private JMenuItem[] menuitem2={new JMenuItem("禁手选择"),new JMenuItem("人机博弈"),new JMenuItem("人人对弈")};private JMenuItem[] menuitem3={new JMenuItem("规则"),new JMenuItem("关于")};private boolean haveai=true;//人与人下还是人与电脑下,true与电脑下Mouseclicked mouseclicked=new Mouseclicked();MouseMoved mousemoved=new MouseMoved();Menuitemclicked menuclicked=new Menuitemclicked();//构造函数public ChessMap(){//改变系统默认字体Font font = new Font("Dialog", Font.PLAIN, 12);java.util.Enumeration keys = UIManager.getDefaults().keys();while (keys.hasMoreElements()) {Object key = keys.nextElement();Object value = UIManager.get(key);if (value instanceof javax.swing.plaf.FontUIResource) {UIManager.put(key, font);}}setTitle("五子棋 ");setSize(FINAL_WIDTH,FINAL_HEIGHT);setResizable(false);init();setLocation(Toolkit.getDefaultToolkit().getScreenSize().width / 2- FINAL_WIDTH / 2, Toolkit.getDefaultToolkit().getScreenSize().height/ 2 - FINAL_HEIGHT / 2);setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);cp.reset();setVisible(true);}//初始化与默认值 public void init()  { map=new ImageIcon(getClass().getResource("bg.jpg"));blackchess=new ImageIcon(getClass().getResource("blackchess.gif"));whitechess=new ImageIcon(getClass().getResource("whitechess.gif"));cp=new ChessPanel(map,blackchess,whitechess);menubar=new JMenuBar();menuitem1[0].setActionCommand("Restart");menuitem1[1].setActionCommand("Rollback");menuitem1[2].setActionCommand("Exit");menuitem2[0].setActionCommand("Forbid");menuitem2[1].setActionCommand("Robot");menuitem2[2].setActionCommand("Human");menuitem3[0].setActionCommand("Rule");menuitem3[1].setActionCommand("About");for(int i=0;i<3;i++)menu[0].add(menuitem1[i]);for(int i=0;i<3;i++)menu[1].add(menuitem2[i]);for(int i=0;i<2;i++)menu[2].add(menuitem3[i]);for(int i=0;i<3;i++)menubar.add(menu[i]);Container p = getContentPane();setJMenuBar(menubar);east = new JPanel();west = new JPanel();p.add(east, "East");p.add(west, "West");p.add(cp, "Center");cp.addMouseListener(mouseclicked);cp.addMouseMotionListener(mousemoved);menuitem1[0].addActionListener(menuclicked);menuitem1[1].addActionListener(menuclicked);menuitem1[2].addActionListener(menuclicked);menuitem2[0].addActionListener(menuclicked);menuitem2[1].addActionListener(menuclicked);menuitem2[2].addActionListener(menuclicked);menuitem3[0].addActionListener(menuclicked);menuitem3[1].addActionListener(menuclicked);}class Mouseclicked extends MouseAdapter//判断鼠标左击并通知棋盘和电脑{public void mouseClicked(MouseEvent e){  if(cp.win==false){    if(haveai){           //和电脑博弈                  Point p1=new Point();                  p1=cp.getPoint(e.getX(),e.getY());                  int x=p1.x;                  int y=p1.y;                          // 如果该位置已经放置棋子                  System.out.println("x="+x+",y="+y);                          if (cp.isChessOn[x][y] != 2)                                     return;                          // 玩家为黑棋,考虑禁手                          if( cp.able_flag && cp.bw == 0) {                              int type = cp.getType(x,y,cp.bw);                              String str = null;                              switch(type){                          case 20:                             str = "黑长连禁手!请选择其它位置下棋!";                            break;                          case 21:                            str = "黑四四禁手!请选择其它位置下棋!";                            break;                          case 22:                             str = "黑三三禁手!请选择其它位置下棋!";                            break;                          default : break;                              }                              if(str != null) {                                 JOptionPane.showMessageDialog(null,str);                                  return;                              }                   }                          boolean flag=cp.haveWin(x, y, cp.bw);                          cp.update( x, y );                          cp.putVoice();  //落子声音                          // 第一步棋,需初始化设置边界值                         if( cp.chess_num == 1){                         if(x-1>=0)                               cp.x_min = x-1;                         if(x-1<=15)                               cp.x_max = x+1;                         if(y-1>=0)                               cp.y_min = y-1;                         if(y-1<=15)                               cp.y_max = y+1;                  }                 else                  cp.resetMaxMin(x,y);                 if (flag) {                     cp.wined(1 - cp.bw);                     return;                 }                 cp.putOne(cp.bw);}else{                                        //和人博弈Point p1=new Point();         p1=cp.getPoint(e.getX(),e.getY());         int x=p1.x;         int y=p1.y;                 // 如果该位置已经放置棋子         System.out.println("x="+x+",y="+y);                 if (cp.isChessOn[x][y] != 2)                             return;                 // 玩家为黑棋,考虑禁手                 if( cp.able_flag && cp.bw == 0) {                     int type = cp.getType(x,y,cp.bw);                     String str = null;                     switch(type){                 case 20:                  str = "黑长连禁手!请选择其它位置下棋!";                 break;                 case 21:                 str = "黑四四禁手!请选择其它位置下棋!";                 break;                 case 22:                  str = "黑三三禁手!请选择其它位置下棋!";                 break;                 default : break;                     }                     if(str != null) {                         JOptionPane.showMessageDialog(null,str);                         return;                     }         }                boolean flag=cp.haveWin(x, y, cp.bw);                cp.update( x, y );                cp.putVoice();  //落子声音                cp.repaint();              // 第一步棋,需初始化设置边界值              if( cp.chess_num == 1){                if(x-1>=0)                  cp.x_min = x-1;                  if(x-1<=15)                  cp.x_max = x+1;                  if(y-1>=0)                  cp.y_min = y-1;                  if(y-1<=15)                  cp.y_max = y+1;              }              else               cp.resetMaxMin(x,y);              if (flag) {                  cp.wined(1 - cp.bw);                  return;              }}    } }}class MouseMoved implements MouseMotionListener//调试用,获得鼠标位置{public void mouseMoved(MouseEvent e)    {    cp.showMousePos(e.getPoint());    }    public void mouseDragged(MouseEvent e)    {}}class Menuitemclicked implements ActionListener//菜单消息处理{public void actionPerfORMed(ActionEvent e) {      JMenuItem target = (JMenuItem)e.getSource();      String actionCommand = target.getActionCommand();      if(actionCommand.equals("Restart")){ //重开一局           cp.reset();           if(cp.sbw==cp.WHITE_ONE)           cp.update(7, 7);            //player=cp.BLACK_ONE;      }      if(actionCommand.equals("Rollback")){ //悔棋      if(cp.win) {        JOptionPane.showMessageDialog(null,"棋局已经结束,不能悔棋!请重新开始新的棋局!");        return;                }        // 当前轮到玩家下棋,取消两步  否则,取消一步        if(cp.chess_num >= 2 && cp.bw == cp.sbw){        cp.isChessOn[cp.pre[cp.chess_num-1][0]][cp.pre[cp.chess_num-1][1]] = 2;        cp.isChessOn[cp.pre[cp.chess_num-2][0]][cp.pre[cp.chess_num-2][1]] = 2;        cp.chess_num -= 2;        cp.repaint();        }        else if(cp.chess_num >= 1 && cp.bw == 1-cp.sbw){        cp.isChessOn[cp.pre[cp.chess_num-1][0]][cp.pre[cp.chess_num-1][1]] = 2;        cp.chess_num --;       cp.repaint();       }      }      else if(actionCommand.equals("Exit")){ //退出        System.exit(1);      }      else if(actionCommand.equals("Forbid")){     //禁手选择        Object[] options = { "无禁手", "有禁手" };        int sel = JOptionPane.showOptionDialog(          null, "你的选择:", "禁手选择",          JOptionPane.DEFAULT_OPTION,          JOptionPane.QUESTION_MESSAGE, null,          options, options[0]);          if(sel==1){                        cp.able_flag=true;                        System.out.println("有禁手");          }else{              cp.able_flag=false;                        System.out.println("无禁手");          }          }      else if(actionCommand.equals("Robot")){            //人机博弈      haveai=true;      Object[] options = { "人类先手", "机器先手" };        int sel = JOptionPane.showOptionDialog(          null, "你的选择:", "先手选择",          JOptionPane.DEFAULT_OPTION,          JOptionPane.QUESTION_MESSAGE, null,          options, options[0]);          if(sel==1){       //机器先手              cp.sbw=cp.WHITE_ONE;              cp.update(7, 7);              System.out.println("机器先手");                       }else{             //人先手              //player=cp.BLACK_ONE;              cp.sbw=cp.BLACK_ONE;              System.out.println("人先手");          }      }          else if(actionCommand.equals("Human")){ //人人博弈        haveai=false;        cp.setHumanhuman(true);      }else if(actionCommand.equals("Rule")){          //规则      JOptionPane.showConfirmDialog(null,      "1、无禁手:" +"\n"+"   黑白双方依次落子,任一方先在棋盘上形成连续的五个(含五个以上)棋子的一方为胜。" +"\n"+"2、有禁手:(走禁手就输,禁手不能落子)" +"\n"+"   鉴于无禁手规则黑棋必胜,人们不断采用一些方法限制黑棋先行的优势,以平衡黑白双方的形式。" +"\n"+"   于是针对黑棋的各种禁手逐渐形成。" +"\n"+"   禁手主要分为以下几类:" +"\n"+"   (1)黑长连禁手:连成六个以上连续相同的棋子。" +"\n"+"   (2)黑三三禁手:两个以上的活三。" + "\n"+"   (3)黑四四禁手:两个以上的四。" + "\n"+"   禁手是针对黑棋而言的,白棋没有任何禁手。" ,"规则",JOptionPane.CLOSED_OPTION,JOptionPane.INFORMATION_MESSAGE);      }      else if(actionCommand.equals("About")){ //版权与帮助        JOptionPane.showConfirmDialog(null,"团队成员:\n" +"自行添加","关于",JOptionPane.CLOSED_OPTION,JOptionPane.INFORMATION_MESSAGE);      }    }}  public static void main(String[] args) {    new ChessMap();  }}

ChessPanel.java类

import javax.sound.sampled.AudioInputStream;import javax.swing.*;import java.applet.Audioclip;import java.awt.*;import java.net.URL;import java.util.Arrays;import java.util.LinkedList;import java.util.Random;@SuppressWarnings("serial")public class ChessPanel extends JPanel{private ImageIcon map;//棋盘背景位图  private ImageIcon blackchess;//黑子位图  private ImageIcon whitechess;//白子位图  public int isChessOn [][];//棋局    protected boolean win = false;          // 是否已经分出胜负    protected int win_bw;                   // 胜利棋色    protected int deep = 3, weight = 7;    // 搜索的深度以及广度    public int drawn_num = 110;           // 和棋步数    int chess_num = 0;                      // 总落子数目    public int[][] pre = new int[drawn_num + 1][2];    // 记录下棋点的x,y坐标   最多 (drawn_num + 1) 个    public int sbw = 0;                          //玩家棋色黑色0,白色1    public int bw = 0;                           // 当前应该下的棋色  0:黑色(默认), 1:白色    // 边界值,用于速度优化    protected int x_max = 15, x_min = 0;    protected int y_max = 15, y_min = 0;    protected boolean able_flag = true;       // 是否选择禁手标志 0:无禁手  1:有禁手(默认  private int h;//棋子长 private int w;//棋子宽 private int insx;//插入棋子的位置 private int insy; private Point mousePoint;//鼠标当前位置 private int winer;//获胜方    private boolean humanhuman=false;       //是否是人人对弈 private int plast=0;//走了几步了, public int BLACK_ONE;//0表黑子 public int WHITE_ONE;//1表白子 public int NONE_ONE;//2表无子 public int N;//棋盘边长  //-------声音  String[] choics = { "put.wav", "win.wav","lost.wav" }; //声音文件名数组  URL file1 = getClass().getResource(choics[0]); //落子声音文件  URL file2 = getClass().getResource(choics[1]); //获胜声音文件  URL file3 = getClass().getResource(choics[2]); //失败声音文件  AudioClip soundPut = java.applet.Applet.newAudioClip(file1); //落子声音剪辑对象  AudioClip soundWin = java.applet.Applet.newAudioClip(file2); //获胜声音剪辑对象  AudioClip soundLost = java.applet.Applet.newAudioClip(file3); //失败声音剪辑对象  public ChessPanel(){} public ChessPanel(ImageIcon r_map,ImageIcon r_blackchess,ImageIcon r_whitechess) {  N=15; map=new ImageIcon(); blackchess=new ImageIcon(); whitechess=new ImageIcon();  map=r_map;  blackchess=r_blackchess;  whitechess=r_whitechess;  NONE_ONE=2;  BLACK_ONE=0;  WHITE_ONE=1;  winer=NONE_ONE;   isChessOn=new int[N][N];    h=blackchess.getIconHeight()*(N-1);    w=blackchess.getIconWidth()*(N-1);    insx=0;    insy=0;    mousePoint=new Point();        }    public void reset(){//重开一局  winer=NONE_ONE;  for(int i=0;i<N;i++)  for(int j=0;j<N;j++){  isChessOn[i][j]=NONE_ONE;  }  chess_num = 0;    win = false;   win_bw=2;  bw = 0;  x_max = 15; x_min = 0;      y_max = 15;y_min = 0;  repaint();    }    public void showMousePos(Point p){//调试用,显示鼠标位置      int cw;      cw=h/N;      mousePoint.x=p.x/cw;      mousePoint.y=p.y/cw;      repaint();    }    public Point getPoint(int x,int y){    int cw;      insx=x;      insy=y;      cw=h/N;    Point r=new Point(x/cw,y/cw);    return r;    }  public void gameOver(int r_winer){//游戏胜负已分  winer=r_winer;  }  public void paint(Graphics g){//整体布局    super.paint(g);    paintChessMap(g);     paintChess(g);    if(winer==BLACK_ONE){    g.drawString(new String("游戏结束!黑棋获胜!"),500,200);        }    else if(winer==WHITE_ONE){    g.drawString(new String("游戏结束!白棋获胜!"),500,200);    }  }  private void paintChessMap(Graphics g){//画棋盘  map.paintIcon(this,g,10,10);  int j;    g.setColor(Color.BLACK);    for(j=0;j<N;j++){//画线    g.drawLine(h/N/2,h/N*j+h/N/2,w-w/N+(N%2)*(h/N/2),h/N*j+h/N/2);    g.drawLine(w/N*j+h/N/2,h/N/2,w/N*j+h/N/2,h-h/N+(N%2)*(h/N/2));    }    g.fillRect(w/N*7+h/N/2-3,h/N*7+h/N/2-3,6,6);//画5个黑方块    g.fillRect(w/N*3+h/N/2-3,h/N*3+h/N/2-3,6,6);    g.fillRect(w/N*11+h/N/2-3,h/N*3+h/N/2-3,6,6);    g.fillRect(w/N*3+h/N/2-3,h/N*11+h/N/2-3,6,6);    g.fillRect(w/N*11+h/N/2-3,h/N*11+h/N/2-3,6,6);  }  private void paintChess(Graphics g){//画棋子  int i,j;  for(i=0;i<N;i++)  for(j=0;j<N;j++){  if(isChessOn[i][j]==BLACK_ONE){  blackchess.paintIcon(this,g,w/N*i,h/N*j);  }  else if(isChessOn[i][j]==WHITE_ONE){  whitechess.paintIcon(this,g,w/N*i,h/N*j);  }  }  }  //-------------------------------下棋声音设置-------------------------------------------------    //落子声音  public void putVoice(){soundPut.play();       }  //获胜声音  public void winVoice(){   soundWin.play();  }  //失败声音  public void lostVoice(){  soundLost.play();  }     //----------------------电脑下棋-------------------------------//  public void  putOne(int bwf ) {  //bwf 棋色 0:黑色 1:白色      int x, y, mx = -100000000;      x = y = -1;      // 搜索最优下棋点      int[][] bests = getBests( bwf );      for (int k = 0; k < bests.length; k++) {          int i = bests[k][0];          int j = bests[k][1];          // 有成5,则直接下子,并退出循环..没有,则思考对方情况          if (getType(i, j, bwf) == 1) {              x = i;              y = j;              break;          }          if (getType(i, j,1 - bwf) == 1) {              x = i;              y = j;              break;          }          // 预存当前边界值          int temp1=x_min,temp2=x_max,temp3=y_min,temp4=y_max;          // 预设己方下棋,并更新边界值          isChessOn[i][j] = bwf;          resetMaxMin(i,j);          // 预测未来          int t = findMin(-100000000, 100000000, deep);          // 还原预设下棋位置以及边界值          isChessOn[i][j] = 2;          x_min=temp1;          x_max=temp2;          y_min=temp3;          y_max=temp4;          // 差距小于1000,50%概率随机选取          //System.out.println("外       :" + i + "," + j + "  mx:" + mx + "  t:" + t);          if (t - mx > 1000 || Math.abs(t - mx)<1000 && randomTest(3)) {              x = i;              y = j;              mx = t;              //System.out.println(i + "," + j + "  mx:" + mx + "  t:" + t);          }               }      System.out.println("x="+x+",y="+y);     // addChess(x,y,(bwf+1)%2,true);     // repaint();      int step=0;step++;System.out.println("step "+step+":-----------------------------------------------");for(int i=0;i<15;i++,System.out.print("\n"))for(int j=0;j<15;j++){if(isChessOn[j][i]!=2)System.out.print(isChessOn[j][i]);elseSystem.out.print(isChessOn[j][i]);}  // 判断是否已分胜负   boolean flag = haveWin(x, y, bwf);       //记录      update( x, y );      repaint();      // 重设边界值      resetMaxMin(x,y);     //  胜负已分      if (flag)           wined(bwf);      if (!flag && chess_num >= drawn_num) {          win = true;          String str = drawn_num + "步没分胜负,判和棋!";          JOptionPane.showMessageDialog(null,str);          return;      }           }    //---------搜索当前搜索状态极大值--------------------------------//  //alpha 祖先节点得到的当前最小最大值,用于alpha 剪枝  //beta  祖先节点得到的当前最大最小值,用于beta 剪枝。  //step  还要搜索的步数  //return 当前搜索子树极大值  protected int findMax(int alpha, int beta, int step) {  int max = alpha;      if (step == 0) {          return evaluate();      }      int[][] rt = getBests(1 - sbw);      for (int i = 0; i < rt.length; i++) {          int x = rt[i][0];      int y = rt[i][1];      if (getType(x, y, 1 - sbw) == 1)   //电脑可取胜      return 100 * ( getMark(1) + step*1000 );          isChessOn[x][y] = 1 - sbw;          // 预存当前边界值          int temp1=x_min,temp2=x_max,temp3=y_min,temp4=y_max;          resetMaxMin(x,y);          int t = findMin(max, beta, step - 1);          isChessOn[x][y] = 2;          // 还原预设边界值          x_min=temp1;          x_max=temp2;          y_min=temp3;          y_max=temp4;          if (t > max)          max = t;          //beta 剪枝          if (max >= beta)               return max;      }      return max;  }     //-----------------------搜索当前搜索状态极小值---------------------------------//   //alpha 祖先节点得到的当前最小最大值,用于alpha 剪枝  //beta  祖先节点得到的当前最大最小值,用于beta 剪枝  //step  还要搜索的步数 //return 当前搜索子树极小值。  protected int findMin(int alpha, int beta, int step) {  int min = beta;      if (step == 0) {          return evaluate();      }      int[][] rt = getBests(sbw);      for (int i = 0; i < rt.length; i++) {          int x = rt[i][0];          int y = rt[i][1];          int type = getType(x, y, sbw);          if (type == 1)       //玩家成5              return -100 * ( getMark(1) + step*1000 );          // 预存当前边界值          int temp1=x_min,temp2=x_max,temp3=y_min,temp4=y_max;          isChessOn[x][y] = sbw;          resetMaxMin(x,y);          int t = findMax( alpha, min, step - 1 );          isChessOn[x][y] = 2;          // 还原预设边界值          x_min=temp1;          x_max=temp2;          y_min=temp3;          y_max=temp4;          if (t < min)          min = t;          //alpha 剪枝          if (min <= alpha) {              return min;          }      }      return min;  }   //-----------------选取局部最优的几个落子点作为下一次扩展的节点---------//   //bwf 棋色 0:黑棋 1:白棋   //return 选出来的节点坐标  private int[][] getBests(int bwf) {      int i_min=(x_min==0 ? x_min:x_min-1);      int j_min=(y_min==0 ? y_min:y_min-1);      int i_max=(x_max==15 ? x_max:x_max+1);      int j_max=(y_max==15 ? y_max:y_max+1);      int n = 0;      int type_1,type_2;      int[][] rt = new int[(i_max-i_min) * (j_max-j_min)][3];      for ( int i = i_min; i < i_max; i++)       for (int j = j_min; j < j_max; j++)      if (isChessOn[i][j] == 2) {                  type_1 = getType(i, j, bwf);                  type_2 = getType(i, j, 1 - bwf);                  if(able_flag && bwf==0 && (type_1 == 20 || type_1 == 21 || type_1 == 22)) // 禁手棋位置,不记录                  continue;                  rt[n][0] = i;                  rt[n][1] = j;                  rt[n][2] = getMark(type_1) + getMark(type_2);                  n++;      }      // 对二维数组排序      Arrays.sort(rt, new ArrComparator());      int size = weight > n? n:weight;      int[][] bests = new int[size][3];      System.arraycopy(rt, 0, bests, 0, size);      return bests;  }   //----------------------------计算指定方位上的棋型-------------------//   // x,y 方向线基准一点。   //ex,ey 指定方向步进向量。   // k 棋子颜色,0:黑色,1:白色   // 该方向上的棋子数目 以及 活度  private int[] count(int x, int y, int ex, int ey, int bwf) {  // 该方向没意义,返回0      if( !makesense(x, y, ex, ey, bwf))          return new int[] {0, 1};            // 正方向 以及 反方向棋子个数  int rt_1 = 1,rt_2 = 1;  // 总棋子个数  int rt = 1;  // 正方向 以及 反方向连子的活度      int ok_1 = 0,ok_2 =0;      // 总活度      int ok = 0;      // 连子中间有无空格      boolean flag_mid1 =false,flag_mid2 = false;      // 连子中间空格的位置      int flag_i1 = 1,flag_i2 = 1;            if (isChessOn[x][y] != 2) {          throw new IllegalArgumentException("position x,y must be empty!..");      }      int i;      // 往正方向搜索      for (i = 1; x + i * ex < 15 && x + i * ex >= 0 && y + i * ey < 15 && y + i * ey >= 0; i++) {          if (isChessOn[x + i * ex][y + i * ey] == bwf)              rt_1++;      // 位置为空,若中空标志为false,则记为中空并继续搜索  否则,break          else if(isChessOn[x + i * ex][y + i * ey] == 2) {          if(!flag_mid1) {          flag_mid1 = true;          flag_i1 = i;          }          else           break;          }          // 位置为对方棋子          else              break;      }      // 计算正方向活度,,      // 最后一个位置不超过边界      if (x + i * ex < 15 && x + i * ex >= 0 && y + i * ey < 15 && y + i * ey >= 0) {      // 最后一个位置为空位 +1活      if( isChessOn[x + i * ex][y + i * ey] == 2) {      ok_1++;      // 若是在尾部检测到连续的空格而退出搜索,则不算有中空              if(rt_1 == flag_i1)      flag_mid1 = false;              // 若中空的位置在4以下 且 棋子数>=4,则这一边的4非活              if(flag_mid1 && rt_1 > 3 && flag_i1 < 4) {              ok_1--;              }      }      // 最后一个位置不是空格,且搜索了2步以上,若前一个是空格,  则不算中空,且为活的边      else if( isChessOn[x + i * ex][y + i * ey] != bwf && i >= 2)           if(isChessOn[x + (i-1) * ex][y + (i-1) * ey] == 2) {          ok_1++;          flag_mid1 = false;          }      }      // 最后一个位置是边界  搜索了2步以上,且前一个是空格,  则不算中空,且为活的边      else if(i >= 2 && isChessOn[x + (i-1) * ex][y + (i-1) * ey] == 2) {      ok_1++;      flag_mid1 = false;      }            // 往反方向搜索              for (i = 1; x - i * ex >= 0 && x - i * ex < 15 && y - i * ey >= 0 && y - i * ey < 15; i++) {          if (isChessOn[x - i * ex][y - i * ey] == bwf)              rt_2++;          else if(isChessOn[x - i * ex][y - i * ey] == 2) {          if(!flag_mid2) {          flag_mid2 = true;          flag_i2 = i;          }          else          break;          }          else              break;      }      // 计算反方向活度      if (x - i * ex < 15 && x - i * ex >= 0 && y - i * ey < 15 && y - i * ey >= 0) {      if( isChessOn[x - i * ex][y - i * ey] == 2) {      ok_2++;      if(rt_2 == flag_i2)      flag_mid2 = false;          if(flag_mid2 && rt_2 > 3 && flag_i2 < 4) {              ok_2--;              }      }      else if( isChessOn[x - i * ex][y - i * ey] != bwf && i >= 2 )       if(isChessOn[x - (i-1) * ex][y - (i-1) * ey] == 2) {      ok_2++;      flag_mid2 = false;      }      }      else if(i >= 2 && isChessOn[x - (i-1) * ex][y - (i-1) * ey] == 2) {      ok_2++;  flag_mid2 = false;      }            //------------------分析棋子类型      // 两边都没中空,直接合成      if( !flag_mid1 && !flag_mid2 ) {      rt = rt_1 + rt_2 - 1;      ok = ok_1 + ok_2;      return new int[] {rt, ok};      }      // 两边都有中空      else if( flag_mid1 && flag_mid2 ){      int temp = flag_i1 + flag_i2 - 1;      // 判断中间的纯连子数,在5以上,直接返回;  为4,返回活4;        if(temp >= 5)      return new int[] {temp, 2};      if(temp == 4)       return new int[] {temp, 2};      // 先看有没死4,再看有没活3,剩下只能是死3      if(rt_1 + flag_i2 - 1 >= 4 || rt_2 + flag_i1 - 1 >= 4)       return new int[] {4, 1};      if(rt_1+flag_i2-1 == 3 && ok_1 > 0 || rt_2+flag_i1-1 == 3 && ok_2 > 0)      return new int[] {3, 2};      return new int[] {3, 1};      }      // 有一边有中空      else {      // 总棋子数少于5,直接合成      if( rt_1 + rt_2 - 1 < 5 )      return new int[] {rt_1 + rt_2 - 1, ok_1 + ok_2};      // 多于5,先找成5,再找活4,剩下的只能是死4      else {      if(flag_mid1 && rt_2 + flag_i1 - 1 >= 5)       return new int[] {rt_2 + flag_i1 - 1, ok_2 + 1};      if(flag_mid2 && rt_1 + flag_i2 - 1 >= 5)       return new int[] {rt_1 + flag_i2 - 1, ok_1 + 1};            if(flag_mid1 && (rt_2 + flag_i1 - 1 == 4 && ok_2 == 1 || flag_i1 == 4) )      return new int[] {4, 2};      if(flag_mid2 && (rt_1 + flag_i2 - 1 == 4 && ok_1 == 1 || flag_i2 == 4) )      return new int[] {4, 2};            return new int[] {4, 1};      }      }  }   //----------------------------判断指定方向下棋是否有意义,即最大可能的棋子数是否 >=5-------------------------------//   // x,y 评估的基准点   // ex,ey 方向向量   // k 棋色   // true:有意义 false:没意义  private Boolean makesense(int x, int y, int ex, int ey, int bwf) {      int rt = 1;      for (int i = 1; x + i * ex < 15 && x + i * ex >= 0 && y + i * ey < 15 && y + i * ey >= 0 && rt < 5; i++)          if (isChessOn[x + i * ex][y + i * ey] != 1 - bwf)              rt++;          else              break;      for (int i = 1; x - i * ex >= 0 && x - i * ex < 15 && y - i * ey >= 0 && y - i * ey < 15 && rt < 5; i++)          if (isChessOn[x - i * ex][y - i * ey] != 1 - bwf)              rt++;          else              break;      return (rt >= 5);  }   //------------------------------------ 棋型判别-------------------------------------//   // x,y 落子位置   // bwf 棋色  0:黑子,1:白子   // 对应的棋型: 棋型代码对应如下:   //             1:成5   //             2:成活4或者是双死4或者是死4活3   //             3:成双活3   //             4:成死3活3   //             5:成死4   //             6:单活3   //             7:成双活2  //             8:成死3   //            9:成死2活2   //            10:成活2   //             11:成死2   //             12: 其他   //             20: 长连禁手   //             21: 双四禁手   //            22: 双活三禁手  protected int getType(int x, int y, int bwf) {  if (isChessOn[x][y] != 2)          return -1;  int[][] types = new int[4][2];  types[0] = count(x, y, 0, 1, bwf);   // 竖直      types[1] = count(x, y, 1, 0, bwf);   // 横向      types[2] = count(x, y, -1, 1, bwf);  // 斜上      types[3] = count(x, y, 1, 1, bwf);   // 斜下      // 各种棋型的方向的数目      int longfive = 0;      int five_OR_more = 0;      int four_died = 0, four_live = 0;      int three_died = 0, three_live = 0;      int two_died  = 0, two_live = 0;      // 各方向上棋型的判别      for (int k = 0; k < 4; k++) {      if (types[k][0] > 5) {        longfive++;              // 长连      five_OR_more++;      }      else if (types[k][0] == 5)      five_OR_more++;          // 成5          else if (types[k][0] == 4 && types[k][1] == 2)          four_live++;             // 活4          else if (types[k][0] == 4 && types[k][1] != 2)          four_died++;             // 死4          else if (types[k][0] == 3 && types[k][1] == 2)          three_live ++;           // 活3          else if (types[k][0] == 3 && types[k][1] != 2)          three_died++;            // 死3          else if (types[k][0] == 2 && types[k][1] == 2)          two_live++;              // 活2          else if (types[k][0] == 2 && types[k][1] != 2)          two_died++;              // 死2          else              ;      }      // 总棋型的判别      if(bwf == 0 && able_flag) {  // 黑棋且选择有禁手      if (longfive != 0)        // 长连禁手      return 20;      if (four_live + four_died >=2)  // 双4禁手      return 21;      if (three_live  >=2)        // 双活三禁手      return 22;      }      if (five_OR_more != 0)          return 1;   // 成5      if (four_live != 0 || four_died >= 2 || four_died != 0 && three_live  != 0)          return 2;   // 成活4或者是双死4或者是死4活3      if (three_live  >= 2)          return 3;   // 成双活3      if (three_died != 0 && three_live  != 0)          return 4;   // 成死3活3      if (four_died != 0)          return 5;   // 成死4      if (three_live  != 0)          return 6;   // 单活3      if (two_live >= 2)          return 7;   // 成双活2      if (three_died != 0)          return 8;   // 成死3      if (two_live != 0 && two_died != 0)          return 9;   // 成死2活2      if (two_live != 0)          return 10;  // 成活2      if (two_died != 0)          return 11;  // 成死2      return 12;  }   //--------------------------对当前棋面进行打分------------------------------------------------------------//  protected int evaluate() {  int rt = 0, mt_c = 1, mt_m = 1;  if(bw == sbw)  mt_m = 2;  else  mt_c = 2;  int i_min=(x_min==0 ? x_min:x_min-1);      int j_min=(y_min==0 ? y_min:y_min-1);      int i_max=(x_max==15 ? x_max:x_max+1);      int j_max=(y_max==15 ? y_max:y_max+1);      for (int i = i_min; i < i_max; i++)          for (int j = j_min; j < j_max; j++)              if (isChessOn[i][j] == 2) {              // 电脑棋面分数                  int type = getType(i, j, 1 - sbw );                  if(type == 1)      // 棋型1,棋型2以及棋型3,加权.  防止"4个双活3"的局分大于"1个双四"之类的错误出现                  rt += 30 * mt_c * getMark(type);                  else if(type == 2)                  rt += 10 * mt_c * getMark(type);                  else if(type == 3)                  rt += 3 * mt_c * getMark(type);                  else                  rt += mt_c * getMark(type);                  // 玩家棋面分数                  type = getType(i, j, sbw );                  if(type == 1)                  rt -= 30 * mt_m * getMark(type);                  else if(type == 2)                  rt -= 10 * mt_m * getMark(type);                  else if(type == 3)                  rt -= 3 * mt_m * getMark(type);                  else                  rt -= mt_m * getMark(type);              }      return rt;  }   //--------------------------------下棋后,更新信息-----------------------------//  void update(int x,int y) {  isChessOn[x][y] = bw;      bw = 1 - bw;      pre[chess_num][0] = x;      pre[chess_num][1] = y;      chess_num++;  }     //-------------------------------------- 下棋后,重设边界值------------------------------//   // x 当前下棋位置的x坐标   // y 当前下棋位置的y坐标  public void resetMaxMin(int x,int y){if(x-1>=0)      x_min = (x_min<x-1 ? x_min:x-1);      if(x+1<=15)      x_max = (x_max>x+1 ? x_max:x+1);      if(y-1>=0)      y_min = (y_min<y-1 ? y_min:y-1);      if(y+1<=15)      y_max = (y_max>y+1 ? y_max:y+1);    }     //------------------------------------------对分数相同的落子点,随机选取-------------------//   //   kt 随机因子 值越小,被选取的概率越大   //  return 是否选择该位置  private boolean randomTest(int kt) {      Random rm = new Random();      return rm.nextInt() % kt == 0;  }   //------------------------------------- 不同棋型对应分数---------------------------------   // k 棋型代号   //return 对应分数  private int getMark(int k) {      switch (k) {      case 1:                             return 100000;      case 2:                             return 30000;      case 3:          return 5000;      case 4:          return 1000;      case 5:          return 500;      case 6:          return 200;      case 7:          return 100;      case 8:          return 50;      case 9:          return 10;      case 10:          return 5;      case 11:          return 3;      case 12:         return 2;      default:                     //禁手棋型          return 0;      }  }   //--------------------------------------- 判断是否已分出胜负---------------------------------------------   // x 落子点x坐标    y 落子点y坐标   // bwf 棋色 0:黑色 1:白色   // return true:分出胜负 false:未分出胜负  public boolean haveWin(int x, int y, int bwf) {      boolean flag = false;      if (count(x, y, 1, 0, bwf)[0] >= 5)          flag = true;      if (!flag && count(x, y, 0, 1, bwf)[0] >= 5)          flag = true;      if (!flag && count(x, y, 1, 0, bwf)[0] >= 5)          flag = true;      if (!flag && count(x, y, 1, -1, bwf)[0] >= 5)          flag = true;      if (!flag && count(x, y, 1, 1, bwf)[0] >= 5)          flag = true;      // 测试用,激活此行代码,不会有输赢..   flag = false;      return flag;  }  public void wined(int bw) {  boolean hh=getHumanhuman();  if(!hh){           //不是人人对弈         win = true;           win_bw = bw;           String str = (bw == sbw ? "恭喜!你赢了!" : "电脑赢了,你还要继续努力啊!");           if(bw==sbw)            winVoice();           else            lostVoice();           JOptionPane.showMessageDialog(null,str);  }  else{             //人人对弈  win = true;          win_bw = bw;          String str = (bw == BLACK_ONE ? "恭喜!黑棋获胜!" : "恭喜!白棋获胜!");         winVoice();          JOptionPane.showMessageDialog(null,str);  }  }public void setHumanhuman(boolean humanhuman) {this.humanhuman = humanhuman;}public boolean getHumanhuman() {return humanhuman;}}

效果图展示

Java怎么实现简单的五子棋游戏

读到这里,这篇“Java怎么实现简单的五子棋游戏”文章已经介绍完毕,想要掌握这篇文章的知识点还需要大家自己动手实践使用过才能领会,如果想了解更多相关内容的文章,欢迎关注编程网精选频道。

--结束END--

本文标题: Java怎么实现简单的五子棋游戏

本文链接: https://www.lsjlt.com/news/329321.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • java实现简单的五子棋游戏
    目录一、主要界面二、功能概况三、代码部分四、部分效果展示本文实例为大家分享了java实现简单五子棋游戏的具体代码,供大家参考,具体内容如下 一、主要界面 1、登录界面;2、游戏选择界...
    99+
    2024-04-02
  • Java怎么实现简单的五子棋游戏
    本文小编为大家详细介绍“Java怎么实现简单的五子棋游戏”,内容详细,步骤清晰,细节处理妥当,希望这篇“Java怎么实现简单的五子棋游戏”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。项目结构这个是在网上找的资源,...
    99+
    2023-06-30
  • java怎么实现简单五子棋小游戏
    本篇文章为大家展示了java怎么实现简单五子棋小游戏,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。讲解五子棋,实际上就是用一个数组来实现的。没有其他很复杂的结构。首先我们制作五子棋,先要有一个棋盘。...
    99+
    2023-06-26
  • java swing怎么实现简单的五子棋游戏
    这篇文章将为大家详细讲解有关java swing怎么实现简单的五子棋游戏,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。用java swing写的一个简单的五子棋游戏。下面是Main.java。packag...
    99+
    2023-06-06
  • java实现简单五子棋小游戏(2)
    本文实例为大家分享了java实现简单五子棋小游戏游戏的具体代码,供大家参考,具体内容如下 讲解 在第一步实现的基础上,添加游戏结束条件。五子棋游戏中的相同棋子如果同时有五个连接成一条...
    99+
    2024-04-02
  • java实现简单五子棋小游戏(1)
    本文实例为大家分享了java实现简单五子棋小游戏的具体代码,供大家参考,具体内容如下 讲解 五子棋,实际上就是用一个数组来实现的。没有其他很复杂的结构。首先我们制作五子棋,先要有一个...
    99+
    2024-04-02
  • jQuery实现简单五子棋游戏
    本文实例为大家分享了jQuery实现五子棋游戏的具体代码,供大家参考,具体内容如下 五子棋小游戏实现的基本思路: 以15*15标准面板为棋盘布局,黑白棋子交替下棋。每落下一枚棋子就判...
    99+
    2024-04-02
  • QT实现简单五子棋游戏
    本文实例为大家分享了QT实现简单五子棋游戏的具体代码,供大家参考,具体内容如下 FIR.pro #----------------------------------------...
    99+
    2024-04-02
  • pygame实现简单五子棋游戏
    本文实例为大家分享了pygame实现简单五子棋游戏的具体代码,供大家参考,具体内容如下 看代码: ①Gomuku2.py: import sys import random impo...
    99+
    2024-04-02
  • python实现简单五子棋小游戏
    用python实现五子棋简单人机模式的练习过程,供大家参考,具体内容如下 最近在初学python,今天就用自己的一些粗浅理解,来记录一下这几天的python简单人机五子棋游戏的练习,...
    99+
    2024-04-02
  • java实现简易的五子棋游戏
    本文实例为大家分享了java实现简易五子棋游戏的具体代码,供大家参考,具体内容如下 先上效果图 一、问题分析 1、五子棋游戏分析: 五子棋作为较为普遍且简易的娱乐游戏,受到众多人的...
    99+
    2024-04-02
  • Java实现简易五子棋小游戏
    本文实例为大家分享了Java实现简易五子棋小游戏的具体代码,供大家参考,具体内容如下 五子棋是一个简单小游戏,首先我们先想想五子棋都有什么东西,棋子,棋盘; 首先我们可以定义一个棋子...
    99+
    2024-04-02
  • Java怎么实现简易五子棋小游戏
    这篇文章主要介绍“Java怎么实现简易五子棋小游戏”,在日常操作中,相信很多人在Java怎么实现简易五子棋小游戏问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Java怎么实现简易五子棋小游戏”的疑惑有所帮助!...
    99+
    2023-06-30
  • Java实现简单的五子棋游戏示例代码
    目录项目结构核心代码ArrComparator.java类ChessMap.java类ChessPanel.java类效果图展示项目结构 这个是在网上找的资源,出处记不得了,记录一下...
    99+
    2024-04-02
  • C语言实现简单五子棋游戏
    在生活中五子棋是一个十分普遍的一款游戏,今天让我们一起来实现这款游戏。 1.初始化棋盘 char ret; //数据存储在一个二维数组中,玩家下去*,电脑下去#. char ...
    99+
    2024-04-02
  • 原生JavaScript实现简单五子棋游戏
    本文实例为大家分享了JavaScript实现简单五子棋游戏的具体代码,供大家参考,具体内容如下 HTML页面 注释都很明确了,大家好好学习。 <!DOCTYPE html&...
    99+
    2024-04-02
  • JavaScript实现简单五子棋游戏的方法
    本篇内容介绍了“JavaScript实现简单五子棋游戏的方法”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!本文实例为大家分享了JavaScr...
    99+
    2023-06-20
  • C语言实现简单的五子棋游戏
    本文实例为大家分享了c语言实现简单五子棋游戏的具体代码,供大家参考,具体内容如下 环境vs2017 一、游戏设计思想 1.该代码设置为 玩家1(*) vs 玩家2(O) 2.选择玩游...
    99+
    2024-04-02
  • Java实现五子棋游戏
    本文实例为大家分享了Java实现五子棋游戏的具体代码,供大家参考,具体内容如下 一、功能分析 五子棋的实现还是较为简单的,通过下期的流程我们可以知道大概要实现一下功能: 1、格界面 ...
    99+
    2024-04-02
  • java实现简单三子棋游戏
    本文实例为大家分享了java实现简单三子棋游戏的具体代码,供大家参考,具体内容如下 JOptionPane类的使用 编写程序,实现简单的三子棋游戏。在三子棋中,双方在3×...
    99+
    2024-04-02
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作