Java實現五子棋遊戲
本文實例為大傢分享瞭Java實現五子棋遊戲的具體代碼,供大傢參考,具體內容如下
一、功能分析
五子棋的實現還是較為簡單的,通過下期的流程我們可以知道大概要實現一下功能:
1、格界面
2、點擊下棋
3、悔棋
4、判斷輸贏
二、功能實現
根據之前的功能分析,要有網格先要有窗體,我們先重新寫一個類,來繼承JFrame類,以便在窗口變動的情況下,對窗口進行重繪(防止在窗口大小發生改變的時候,之前的繪畫會消失),這裡我們重寫paint方法,畫出網格線
public class MyFrame extends JFrame{ public void paint(Graphics g) { super.paint(g); //重畫格子 for (int i = 0; i < 700; i = i+50) { g.drawLine(100,100+i,750,100+i); g.drawLine(100+i,100,100+i,750); } }
然後通過簡單的設置窗口信息來把窗口實現,同時,在窗體內加好開始按鈕和悔棋按鈕,為窗體添加鼠標監聽器和鼠標移動監聽器,為按鈕添加事件監聽器,為下一步實現下棋晦氣功能做準備。
public class Chess { public static void main(String[] args) { Chess chess=new Chess(); chess.ChessUI(); } public void ChessUI() { MyFrame jf=new MyFrame(); jf.setTitle("Chess"); jf.setSize(900, 900); jf.setLocationRelativeTo(null); jf.setBackground(Color.WHITE); jf.setDefaultCloseOperation(3); FlowLayout border=new FlowLayout(); jf.setLayout(border); Dimension dmb = new Dimension(80,30); JButton startGame = new JButton("Start"); startGame.setSize(dmb); jf.add(startGame); JButton retract = new JButton("Retract"); retract.setSize(dmb); jf.add(retract); Buttonlistener bl = new Buttonlistener(); jf.addMouseListener(bl); jf.addMouseMotionListener(bl); startGame.addActionListener(bl); retract.addActionListener(bl); jf.getContentPane().setBackground(Color.WHITE); jf.setVisible(true); Graphics g=jf.getGraphics(); bl.jf=jf; bl.g=g; // jf.count=bl.count; jf.counts=bl.counts; //傳值與傳址的區別 jf.board=bl.board; } }
接下來就是實現監聽器,監聽器裡要做到下棋、悔棋、判斷輸贏。
下棋的實現是通過點擊下棋按鈕,開始新建一個與網格一樣大小的數組,來存儲棋局。在點擊棋盤的時候獲得鼠標位置,然後通過計算,算出離鼠標最近的網格交點坐標,通過記錄之前的棋局來判斷這個位置能否下棋,通過一個計數器將黑棋or白棋繪在交點上,計算是否有連續的5個同色棋子,彈出輸贏框。
而悔棋則是在點擊悔棋之後,獲得當前計數器的狀態,在記錄棋局的矩陣中,找到上一次子的位置,然後對這個區域進行重繪(先用棋盤色覆蓋棋子,再根據位置重畫該區域的棋盤)
為瞭給用戶更好的交互體驗,我在這裡加入瞭一個位置指示器。這個指示器的作用在於,在點擊開始之後,隨著鼠標的移動,可以在棋盤上顯示如果在當前位置點擊鼠標進行落字,將會落子的位置。這個功能可以通過鼠標移動監聽器來實現,原理同對棋子的繪制,但是要註意的是,在鼠標移動到下一個落字區域的時候,要將之前的指示器用棋盤顏色覆蓋掉。
public class Buttonlistener implements ActionListener,MouseListener,MouseMotionListener{ public Graphics g; public int[][] board = new int [14][14]; public int count=0; public int[] counts=new int[1]; public String s=null; public JFrame jf; public void actionPerformed(ActionEvent e) { s=e.getActionCommand(); if (s=="Start") { for (int i=0;i<board.length;i++) { for (int j=0;j<board[i].length;j++) { board[i][j]=-1; } } } if (s=="Retract") { System.out.print("haha"); for (int m=0;m<14;m++) { for (int n=0;n<14;n++) if (board[m][n]==count) { board[m][n]=-1; count--; counts[0]=count; g.setColor(Color.white); g.fillRect(m*50+100-20, n*50+100-20, 40, 40); g.setColor(Color.black); g.drawLine(m*50+100-20,n*50+100,m*50+100+20,n*50+100); g.drawLine(m*50+100,n*50+100-20,m*50+100,n*50+100+20); s="Start"; break; } } } } public void mouseClicked(MouseEvent e) { if (s=="Start") { int m = Math.round((e.getX()-75)/50); int n = Math.round((e.getY()-75)/50); if (board[m][n] == -1){ count++; counts[0]=count; board[m][n] = count; if (count%2 == 1) { g.setColor(Color.black); g.fillOval(m*50+85, n*50+85, 30, 30); } if (count%2 == 0) { g.setColor(Color.black); g.drawOval(m*50+85, n*50+85, 30, 30); g.setColor(Color.white); g.fillOval(m*50+85, n*50+85, 30, 30); } //判斷輸贏 int chesscount = 1; for (int i=m; i<13 && board[i+1][n]%2==count%2;i++) { chesscount++; } for (int i=m;i>0 && board[i-1][n]%2==count%2;i--) { chesscount++; } if (chesscount==5) { //結束 win(); } else { chesscount=1; for (int i=n; i<13 && board[m][i+1]%2==count%2;i++) { chesscount++; } for (int i=n;i>0 && board[m][i-1]%2==count%2;i--) { chesscount++; } if (chesscount==5) { win(); } else { chesscount=1; for (int i=m,j=n; i<13 && j<13 && board[i+1][j+1]%2==count%2;i++,j++) { chesscount++; } for (int i=m,j=n;i>0 && j>0 && board[i-1][j-1]%2==count%2;i--,j--) { chesscount++; } if (chesscount==5) { win(); } else { chesscount=1; for (int i=m,j=n; i<13 && j>0 && board[i+1][j-1]%2==count%2;i++,j--) { chesscount++; } for (int i=m,j=n;i>0 && j<13 && board[i-1][j+1]%2==count%2;i--,j++) { chesscount++; } if (chesscount==5) { win(); } } } } } } } public void win() { JFrame jf=new JFrame("遊戲結束"); jf.setSize(300, 100); jf.setLocationRelativeTo(null); jf.setDefaultCloseOperation(1); //流式佈局管理器 FlowLayout flow=new FlowLayout(); jf.setLayout(flow); //字樣提示 JTextField jtf=new JTextField(); if (count%2==1) {jtf.setText("黑棋勝出");} else {jtf.setText("白棋勝出");} Dimension dm=new Dimension(60,30); jtf.setSize(dm); jtf.setEditable(false); jf.add(jtf); jf.setVisible(true); } public void mousePressed(MouseEvent e) { } public void mouseReleased(MouseEvent e) { } public void mouseEntered(MouseEvent e) { } public void mouseExited(MouseEvent e) { } public void mouseDragged(MouseEvent e) { } public int m1,n1; public int m2=0,n2=0; public void mouseMoved(MouseEvent e) { m1 = Math.round((e.getX()-75)/50); n1 = Math.round((e.getY()-75)/50); if (s==null) { } if (s=="Start") { if ((m1<14&&m1>=0)&&(n1<14&&n1>=0)&&(board[m1][n1] == -1)&&(m1 != m2 || n1 !=n2)){ g.setColor(Color.WHITE); g.drawLine(m2*50+100-10,n2*50+100-20,m2*50+100-20,n2*50+100-10); g.drawLine(m2*50+100+10,n2*50+100+20,m2*50+100+20,n2*50+100+10); g.drawLine(m2*50+100+20,n2*50+100-10,m2*50+100+10,n2*50+100-20); g.drawLine(m2*50+100-20,n2*50+100+10,m2*50+100-10,n2*50+100+20); m2=m1; n2=n1; } if ((m1<14&&m1>=0)&&(n1<14&&n1>=0)&&(board[m1][n1] == -1)){ g.setColor(Color.red); g.drawLine(m1*50+100-10,n1*50+100-20,m1*50+100-20,n1*50+100-10); g.drawLine(m1*50+100+10,n1*50+100+20,m1*50+100+20,n1*50+100+10); g.drawLine(m1*50+100+20,n1*50+100-10,m1*50+100+10,n1*50+100-20); g.drawLine(m1*50+100-20,n1*50+100+10,m1*50+100-10,n1*50+100+20); } } } } }
在完成瞭這些之後,就可以和朋友下一盤五子棋瞭,但是如果朋友不小心拖動瞭窗口,棋局就不復存在瞭,所以我們要優化一下MyFrame類,通過將存儲棋局的矩陣以及當前的計數器傳入MyFrame類,可以在窗口變化的情況下依舊可以復現棋局。這裡需要註意的是,傳值與傳址的區別。如果簡單的將count傳入MyFrame類,是不能實現將當前計數器傳入的,因為count是一個int類型的數據,這種數據類型是基本數據類型是傳值,而自定義類以及數組等,都是傳址,所以再一次賦值之後,自定義數據類型會實時更新,而基本數據類型則不會再變。為瞭解決這個問題,我們將計數器放到一個數組內,通過地址傳遞傳入MyFrame類的對象內,然後再從數組內取出技術去進行使用。
public class MyFrame extends JFrame{ public int[][] board = new int [14][14]; public int count; public int[] counts=new int[1]; public void paint(Graphics g) { count=counts[0]; super.paint(g); //重畫格子 for (int i = 0; i < 700; i = i+50) { g.drawLine(100,100+i,750,100+i); g.drawLine(100+i,100,100+i,750); } for(;count>0;count--) { System.out.println(""+count); for (int m=0;m<14;m++) { for (int n=0;n<14;n++) if (board[m][n]==count) { if (count%2 == 1) { g.setColor(Color.black); g.fillOval(m*50+85, n*50+85, 30, 30); } if (count%2 == 0) { g.drawOval(m*50+85, n*50+85, 30, 30); g.setColor(Color.white); g.fillOval(m*50+85, n*50+85, 30, 30); } } } } } }
這樣,就可以和朋友安心的進行高手之間的較量瞭。
以上就是本文的全部內容,希望對大傢的學習有所幫助,也希望大傢多多支持WalkonNet。