Java Swing五子棋项目

news2025/1/22 8:08:09

一、项目简介

本项目为Java Swing五子棋项目,主要针对计算机相关专业的正在做毕设的学生与需要项目实战练习的Java学习者。
包含:项目源码、数据库脚本等,该项目附带全部源码可作为毕设使用。
项目都经过严格调试,eclipse 确保可以运行!
该系统功能完善、界面美观、操作简单、功能齐全、管理便捷,具有很高的实际应用价值

二、技术实现

后端:java,面向对象,swing编程和ImageIO类。
运行环境及开发工具:jdk,idea或者eclipse

三、功能描述

Java五子棋项目

该项目主要使用到了swing编程和ImageIO类。

  • 游戏开始:清空棋盘,重新绘图
  • 游戏设置:设置倒计时
  • 游戏说明:说明游戏规则
  • 认输:某一方放弃游戏,另一方直接获胜
  • 引入背景音乐功能:可以引入音频文件
  • 关于:显示程序的作者、版本等
  • 退出:结束程序

##五子棋游戏的功能##

  1. 用户点击鼠标时,将棋子渲染到相应的位置;
  2. 自动判断游戏是否结束。
  3. 对游戏时间进行设置,判断是否超出规定时间。

实现步骤

  1. 渲染游戏界面(swing)。新建一个类FiveCheeseFrame继承JFrame并实现MouseListener接口。在构造方法中初始化界面(大小、位置等)。
  2. 在鼠标事件的mouseClick()方法中响应相关事件。
  3. 为了实现倒计时,需要单独引入一个线程。

该游戏的难点:

1.如何让游戏界面居中显示?

获取屏幕的宽高和游戏的宽高。设置显示位置为屏幕的宽高减去游戏的宽高除以2就行了。

// 取得屏幕的宽高
int screenWidth = Toolkit.getDefaultToolkit().getScreenSize().width;
int screenHeight = Toolkit.getDefaultToolkit().getScreenSize().height;
System.out.println("屏幕的尺寸是:" + screenWidth + "*" + screenHeight);
// 设置游戏界面居中显示,jf是自定义的窗体类,继承自JFrame
jf.setLocation((screenWidth - WIDTH) / 2, (screenHeight - HEIGHT) / 2);

2.在鼠标点击的位置显示棋子?

MouseEvent的getX()getY()方法可以获得点击的位置。黑子可以用实心的黑圆表示,白字可以用空心黑圆+实心白圆表示。Graphics类的drawOval()fillOval()方法。

3.之前下过的棋子的位置如何保存?

通过19*19的二维数组allChess保存每个交叉点的状态(黑子(填充1)、白子(填充2)、无子(填充0))。

4.在窗口中绘制网格和图像

Graphics类,类似画笔,可以在窗口中绘制文字和图像。通过覆写JFrame中的paint()方法来使用,通过repaint()方法(表示重新执行一次paint方法)来调用。

Graphics g;
BufferedImage image;
try {
	image = ImageIO.read(new File("images/title.jpg")); //解决窗体透明问题只需要在窗体中绘制一张图片,在此基础上绘制其他
	g.drawImage(image, 20, 40,this);
} catch (IOException e) {
	e.printStackTrace();
}

5.如何让棋子对齐到十字线?

棋盘绘制完成后,实际上就建立了如下的一个坐标轴为0~18的坐标系坐标系。
对于每一个鼠标点击的点,我们可以取离鼠标点击位置最近的点。关键就是得到0~18的整数。将大坐标转换为小坐标。

思考

每个单元格是25*25像素,我们惠子棋盘的时候采用的是for循环。

for (int i = 0; i < 19; i++) {
	g.drawLine(15, 70 + 25 * i, 465, 70 + 25 * i); // 绘制横线
	g.drawLine(15 + 25 * i, 70, 15 + 25 * i, 520);// 绘制竖线
}

我们通过鼠标监听器得到点击位置的坐标,然后分别减去起始位置,再除以25,就可以得到一个0~18的数,然后通过这个数字,就可以让其精确对准棋盘。

x = e.getX(); // 得到鼠标的点击位置
y = e.getY();

x = (x - 15) / 25; // 得到最近的十字点,可以先得到0~18的整数,然后通过这个整数乘以每行的间距完成
y = (y - 70) / 25;

6.如何判断游戏胜负?

依据:五子棋的游戏规则:是否有同一种颜色的棋子连成5个。判断当前棋子周围是否有5个连续一样棋子——一共需要判断4种情况:水平方向、垂直方向和两条对角线的方向。其中每一种情况又需要判断两次,例如水平方向需要从左向右和从右向左进行判断。
首先得到当前的棋子的颜色color即二维数组allCheese[x][y]

水平方向:y坐标不变,x坐标每次增加1或者减去1.

// 横向判断
int i = 1;
while (color==allChess[x+i][y]) { // 向右判断
	count++;
	i++;
}
i = 1; // i归位,特别注意这里
while (color == allChess[x-i][y]) { // 向左判断
	count++;
	i++;
}
if (count >= 5) {
	flag = true; // flag表示是否有5子相连
}

同理,垂直方向仅仅是x坐标不变,y的坐标每次加1,减1。和上面的算法几乎一样

左上-右下方向:如果判断方向是“左上—>右下”,则x和y的坐标每次都加1;如果从右下判断到左上,则x和y的坐标每次加1,这里同样要注意变量i要复位。

// 左下-右上判断
int i3 = 1;
int count3 = 1;
while (color==allChess[x+i3][y-i3]) { // 左下--->右上---> allChess[x+i3][y-i3]
	count3++;
	i3++;
}
i3 = 1; // i3归位
while (color == allChess[x-i3][y+i3]) { // 右上--->左下---> allChess[x-i3][y+i3]
	count3++;
	i3++;
}
if (count3 >= 5) {
	flag = true;
}

右上-左下方向的判断和左上-右下的判断类似,只是x,y的变化是异号的。

思考

上面的4个方向的算法大体类似,我们可以将其抽象为一个方法。观察到以上4方向上的算法的不同之处就是x坐标和y轴的变化率,同时它们都需要判断当前棋子的颜色color,得到当前相同棋子相连的最大数count,将算法加以抽象得到如下的方法:

/**
 * 得到相同的棋子连接的数量,由上面的4次判断抽象出来的
 * @param xChange:x的变化
 * @param yChange:y的变化
 * @param color:当前棋子状态(黑?白)
 * @return
 */
private int checkCount(int xChange,int yChange,int color){
	
	int count = 1;
	int tempX = xChange, tempY = yChange; // 保存传过来的xChange和yChange,复位的时候需要用到
	// 首先要检查数组下标是否越界
	while (x + xChange >= 0 && x + xChange <= 18 && y + yChange >= 0 && y + yChange <= 18 && color == allChess[x + xChange][y + yChange]) {
		count++;
		if (xChange != 0) {
			xChange++;
		}
		if (yChange != 0) {
			if (yChange > 0) {
				yChange++;
			}else {
				yChange--;
			}
		}
	}
	
	xChange = tempX; // 复位
	yChange = tempY;
	// 首先要检查数组下标是否越界
	while (x - xChange >= 0 && x - xChange <= 18 && y - yChange >= 0 && y - yChange <= 18 && color == allChess[x - xChange][y - yChange]) {
		count++;
		if (xChange != 0) {
			xChange++;
		}
		if (yChange != 0) {
			if (yChange > 0) {
				yChange++;
			}else {
				yChange--;
			}
		}
	}
	return count;
}

得到棋子数只需要这样调用以上的方法:

/* 判断4个方向上的棋子数,判断的顺序不重要 */
count = checkCount(1, 0, color); // 横向的棋子数
if (count >= 5) {
	flag = true;
}else {
	count = checkCount(0, 1, color); // 纵向的棋子数
	if (count >= 5) {
		flag = true;
	}else {
		count = checkCount(1, -1, color); // 左下-右上的棋子数
		if (count >= 5) {
			flag = true;
		}else{
			count = checkCount(1, 1, color); // 左上-右下的棋子数
			if (count >= 5) {
				flag = true;
			}
		}
	}
}

7. 如何设置倒计时##

让窗体类实现Runnable接口,在构造方法中启动线程,并将其挂起。因为游戏刚刚启动的时候并不需要倒计时,只需要用户点击设置游戏时间之后倒计时才开始生效。声明两个全局变量blackTimewhiteTime分别保存黑方和白方额度剩余时间(初始值为0)。用户点击”游戏设置“按钮后,将时间赋值给这两个变量,重新开始游戏(清空棋盘),并将挂起的线程恢复(resume()方法)。在run()方法中首先判断是否有时间的限制,再进一步判断,当前棋子的状态,给对应的时间每隔1秒减去1,当时间变为0的时候,给用户提示,显示输赢信息,并设置不能够再次下棋。需要将时间同步显示到界面。

8.背景音乐的控制

新引入一个背景音乐的线程,在主界面渲染(窗体的构造方法中)的时候启动线程。引入一个全局变量布尔控制背景音乐的开关,当用户点击音乐的时候改变状态,并将背景音乐挂起或者恢复。

四、界面截图

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

五、源码地址

https://download.csdn.net/download/weixin_43860634/87364581

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/141180.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

ArcGIS基础实验操作100例--实验50以栅格分区裁剪面要素

本实验专栏参考自汤国安教授《地理信息系统基础实验操作100例》一书 实验平台&#xff1a;ArcGIS 10.6 实验数据&#xff1a;请访问实验1&#xff08;传送门&#xff09; 高级编辑篇--实验50 以栅格分区裁剪面要素 目录 一、实验背景 二、实验数据 三、实验步骤 &#xff0…

Vulkan PBR与IBL实践

对我来说&#xff0c;每天能过得心情舒畅&#xff0c;有酒喝有美味佳肴吃&#xff0c;必要时工作一会儿&#xff0c;晚上睡得舒舒服服&#xff0c;就行了。 ——迪希亚 序 说实话我已经记不清上一篇文章是什么时候的事情了&#xff0c;感觉得有好几个月了&#xff0c;但其实我…

【Linux】【信号】

文章目录一、信号是什么1.生活中的信号2.什么是Linux信号3.信号处理的常见方式4.Linux当中的信号二、信号的产生1.signal函数2.核心转储3.验证进程等待中的core dump标记位三、信号的系统调用接口1.kill2.raise3.abort四、由软件条件产生信号alarm五、硬件异常产生信号1.除零异…

从编译到可执行,eBPF 加速容器网络的原理分析 | 龙蜥技术

编者按&#xff1a;eBPF(extended Berkeley Packet Filter) 是一种可以在 Linux 内核中运行用户编写的程序&#xff0c;而不需要修改内核代码或加载内核模块的技术。简单说&#xff0c;eBPF 让 Linux 内核变得可编程化了。本文整理自龙蜥大讲堂第 57 期&#xff0c;浪潮信息 SE…

HTML防数据采集

什么是防采集 就是我们想利用爬虫工具采集某个网站的数据&#xff08;前提当然是公开合法数据&#xff09;&#xff0c;但网站不想给你采集而设置的技术阻挡措施。 常见的防止采集方案 利用输入验证码框验证&#xff0c;在采集某些网站过程中&#xff0c;要求你输入验证码&a…

电源特性测试测试哪些方面?电源特性自动测试系统NSAT-8000介绍

假设电源适配器厂家对电源适配器进行了很合理的测试验证工作&#xff0c;那么电源适配器输出的电压应该是个稳定的电源输出。那么对于一些小型设备而言&#xff0c;电源测试就主要测试设备电源端的测试工作。下面纳米软件Namisoft小编将带大家一起看看&#xff0c;关于电源特性…

Android Jetpack Compose——一个简单的笔记APP

一个简单的笔记APP简述效果视频Hilt提供依赖对象Room CRUD接口实现类内容封装查询所有查询删除插入笔记内容效果图ViewModel依赖注入数据初始化数据处理View标题栏排序组件笔记列表新建&编辑笔记效果图ViewModel依赖注入初始化数据处理View背景颜色条标题保存笔记路由导航建…

动态规划 0-1背包问题(滚动数组思想优化)

目录 125 背包问题&#xff08;二&#xff09;LintCode 炼码 0-1背包滚动数组优化 0-1背包问题&#xff08;一&#xff09;LintCode 炼码 【解法一】二维数组 【解法二】滚动数组 125 背包问题&#xff08;二&#xff09;LintCode 炼码 class Solution { public:/*** para…

HADOOP-3.2.2安装

HADOOP-3.2.2安装一. 准备工作二.安装阶段1. 创建安装目录并安装解压包2.修改配置文件core-site.xml3. 修改hdfs-site.xml4. 修改修改yarn-site.xml5.修改workers文件6.修改hadoop-env.sh7.修改mapred-site.xml8.递归创建目录9.分发文件三.运行阶段1.启动hdfs2.启动yarn3.启动j…

F280049C Crossbar X-BAR

文章目录X-BAR9.1 输入X-BAR9.2 ePWM、CLB和GPIO输出X-BAR9.2.1 ePWM X-BAR9.2.1.1 ePWM X-BAR架构9.2.2 CLB X-BAR9.2.2.1 CLB X-BAR架构9.2.3 GPIO输出X-BAR9.2.3.1 GPIO输出X-BAR架构9.2.4 X-BAR标志总结X-BAR 交叉开关&#xff08;在本章中称为X-BAR&#xff09;提供了以各…

一年风雨几度寒,一杯浊酒敬虎年

我是谁大家好&#xff0c;我是凡夫贩夫&#xff0c;真实姓名不值一提&#xff0c;我的履历也很不值一提&#xff0c;非名校非大厂非专家&#xff0c;一名三非野生java开发者&#xff0c;现居住地河南郑州&#xff0c;就职于一家外包公司。的确&#xff0c;我是一个普通人&#…

(02)Cartographer源码无死角解析-(46) 2D栅格地图→CastRay()函数与贝汉明(Bresenham)算法

讲解关于slam一系列文章汇总链接:史上最全slam从零开始&#xff0c;针对于本栏目讲解(02)Cartographer源码无死角解析-链接如下: (02)Cartographer源码无死角解析- (00)目录_最新无死角讲解&#xff1a;https://blog.csdn.net/weixin_43013761/article/details/127350885 文末…

PCB设计完成后,为什么经常要拼版及拼版注意事项

通常我们在完成PCB设计的时候&#xff0c;有一些板子我们通常是需要进行拼版的&#xff0c;那么我们为什么要拼版&#xff0c;哪种情况下需要拼版呢&#xff1f;不拼是否可以呢&#xff1f;1、PCB生产制作尺寸要求 一般来说面积比较小的板子我们是需要进行拼版&#xff0c;一般…

MATLAB APP 设计实践(一)UART通信(下篇)

引言上篇介绍了 MATLAB App 的基本内容&#xff0c;本篇就结合UART发送数据的具体案例介绍开发过程。文末给出设计源文件、设计的可执行文件的下载链接&#xff0c;以及App的实际使用视频&#xff08;与FPGA开发板进行调试验证&#xff09;。前文链接&#xff1a;MATLAB APP 设…

MySQL 分区(innode引擎的讲解)

目录 一.InnoDB逻辑存储结构 段 区 页 二.分区概述 分区 三.分区类型 一.InnoDB逻辑存储结构 首先要先介绍一下InnoDB逻辑存储结构和区的概念&#xff0c;它的所有数据都被逻辑地存放在表空间&#xff0c;表空间又由段&#xff0c;区&#xff0c;页组成。 段 段就是…

【Python】sklearn机器学习之层次聚类算法AgglomerativeClustering

文章目录基本原理绘图层次定义距离基本原理 和Birch聚类相似&#xff0c;层次聚类也是一种依赖树结构实现的聚类方法&#xff0c;其核心概念是相似度。根据相似度&#xff0c;可以将所有样本组织起来&#xff0c;从而构建一棵层次聚类树。 其中Birch算法的核心&#xff0c;叫…

MyBatis【多表查询与动态SQL使用】

MyBatis【多表查询与动态SQL使用】&#x1f34e;一.MyBatis多表查询&#x1f352;1.1 一对一查询&#x1f352;1.2 一对多查询&#x1f34e;二.动态SQL使用&#x1f352;2.1 if 标签使用&#x1f352;2.2 trim 标签使用&#x1f352;2.3 where 标签使用&#x1f352;2.4 set 标…

ARX测试_绘制道路横断面

本文迁移自本人网易博客&#xff0c;写于2011年1月12日&#xff0c;ARX测试_绘制道路横断面 - lysygyy的日志 - 网易博客 (163.com)1、已提供道路的图形&#xff0c;获取用户输入的两点&#xff0c;并在两点间画一条虚线。计算这条直线与多少条直线相交&#xff0c;若数量不等于…

振弦采集模块多通道专用寄存器

振弦采集模块多通道专用寄存器 多通道频率、温度值寄存器 51~58&#xff08; 0x33~0x3A&#xff09; 位 符号 值 描述 默认值 bit15:0 频率/温度值 0 单通道模块时&#xff0c;寄存器 51 内为频率值&#xff0c;寄存器 55 内为温度值 4 通道模块时&#xff0c;寄存器 51~54 内…

如何去学习PMP考试的《PMBOK》

首先&#xff0c;是PMP考试的核心教材&#xff1a;《PMBOK指南》&#xff0c;目前已经出道第七版了&#xff0c;大家如果有备考的需要要赶紧买一本来学习。 其次&#xff0c;是《汪博士解读PMP》&#xff0c;目前出到第6版&#xff0c;这本书是对PMBOK中各领域知识点的深入浅出…