Java实现五子棋游戏
作者:weixin_50869378 发布时间:2022-07-08 12:50:27
本文实例为大家分享了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);
}
}
}
}
}
}
这样,就可以和朋友安心的进行高手之间的较量了。
来源:https://blog.csdn.net/weixin_50869378/article/details/116086234


猜你喜欢
- 目录一、首先导入生成二维码和微信支付环境二、在application.yml文件配置微信所有需的基本配置1.导入2.创建MyWXPayCon
- 一、相关概念1.1 Jenkins概念:Jenkins是一个功能强大的应用程序,允许持续集成和持续交付项目,无论用的是什么平台。这是一个免费
- 本文实例讲述了Android实现仿通讯录侧边栏滑动SiderBar效果代码。分享给大家供大家参考,具体如下:之前看到某些应用的侧边栏做得不错
- 只要是面向对象的编程语言,基本上都有类Class的用法,只是好不好用,好不好记而已,面向对象是c++开始引入的,但是c++ 关于类的东西,弄
- ComparableComparable 是排序接口。若一个类实现了Comparable接口,就意味着“该类支持排序”。 即然实现Compa
- 效果:一个手指实现(所有手势事件)和(部分事件的);A. 所有手势activity_main.xml<TextView android
- 本文实例讲述了Android开发之菜单(menu)用法。分享给大家供大家参考,具体如下:Android手机专门用一个按键“menu“来显示菜
- 微信小程序与aspnetcore signalr实例本文不对小程序与signalr做任何介绍,默认读者已经掌握aspnetcore Sign
- 1. 使用Files.list()迭代目录及其子目录文件Files.list()可以迭代目录及其子目录文件Files.list(Paths.
- 1.设计思路,使用VersionCode定义为版本升级参数。android为我们定义版本提供了2个属性:<manifest packa
- 目录例子1:字符型变量例子2:数据类型转换例子3:使用异或对字符进行加密和解密例子4:短路逻辑或(||)和位运算(|)的区别例子5:用if语
- 一说到写日志,大家可能推荐一堆的开源日志框架,如:Log4Net、NLog,这些日志框架确实也不错,比较强大也比较灵活,但也正因为又强大又灵
- 问题描述最近IDEA抽风了,不管是新建SpringBoot工程,还是导入项目。IDEA代码里面都会飘红~Build项目时,会提示错误:错误:
- JAVA中获取文件MD5值的四种方法其实都很类似,因为核心都是通过JAVA自带的MessageDigest类来实现。获取文件MD5值主要分为
- 我们经常会有这种场景,只需要把Spring Boot打成普通的jar包,不包含配置文件,供其他程序应用本文介绍如何使用Maven将Sprin
- SpringBoot2之PUT请求接收不了参数的解决办法,这个问题,关乎两个Filter过滤器,是spring3和3.5之后提供的,目的就是
- 本文实例为大家分享了Java分页展示的具体代码,供大家参考,具体内容如下先上图吧,大致如图,也就提供个思路(ps:使用了SSH框架) 前台J
- 一.摘要emmmm..对springmvc不太熟练的情况下,如果不出意外的话,项目启动后出现404页面是很烦人。在这里,我记录一下可能会导致
- 介绍在进行项目开发的时候,刚好需要用到对字符串表达式进行求值的处理场景,因此寻找了几个符合要求的第三方组件LambdaParser、Dyna
- 假定你已经了解了运行时的数据区域和常用的垃圾回收算法,也了解了Hotspot支持的垃圾回收器。一、cpu占用过高cpu占用过高要分情况讨论,