五子棋:不会下五子棋也没关系,会用Java写五子棋就行
关注公号“微澜网络”获取完整源代码!
效果展示:
目录
效果展示:
导语:
游戏介绍:
程序设计:
1.游戏规则和功能:
2.用户界面设计:
3.程序架构设计:
4.可扩展性和灵活性:
5.用户体验:
6.测试和优化:
开发步骤:
1.绘制主体框架
2.编写按钮事件类
3.编写棋盘类包含棋子画法
4.重新开始、悔棋功能
(1)重新开始游戏
(2)悔棋
5.项目结构
结语:
导语:
五子棋,一种古老而智慧的棋类游戏,深受广大棋友的喜爱。今天,我们将以一种全新的方式——使用Java编程语言,来探索五子棋的魅力。无论你是否擅长下棋,只要掌握了Java编程,你就能编写出一款属于自己的五子棋游戏。让我们一起来开发一款属于自己的五子棋游戏吧!
大家好,今天用JavaFX技术写一个单机版的五子棋对战小游戏,文中示例思路清晰、代码完整,适合Java初学者尝试实战,供大家参考。
游戏介绍:
五子棋,又称连珠棋,是一种起源于中国古代的棋类游戏。它以两人对弈,在15×15的棋盘上轮流下棋,通过在棋盘上连成五子或五子以上的直线、斜线或对角线来获得胜利。五子棋看似简单,实则蕴含着丰富的战略和战术,需要玩家具备敏锐的观察力、精准的计算能力和出色的布局能力。
程序设计:
1.游戏规则和功能:
-
游戏应该符合五子棋的规则,即两名玩家轮流在棋盘上落子,先在横、竖、斜向连成五子的玩家获胜。
-
提供开始游戏、重新开始、悔棋、退出游戏等功能。
-
实现胜负判断,当一方获胜或棋盘填满平局时结束游戏。
2.用户界面设计:
-
使用图形界面或控制台界面,提供友好的用户交互。
-
显示棋盘、落子情况、玩家信息等。
-
提供按钮或命令行选项以控制游戏流程,如开始游戏、重新开始、悔棋等。
3.程序架构设计:
-
使用面向对象的思想,设计棋盘、玩家、游戏控制器等类。
-
实现游戏逻辑与界面的分离,便于后续维护和扩展。
-
使用合适的数据结构来表示棋盘状态,例如二维数组。
4.可扩展性和灵活性:
-
考虑到后续可能的需求变更,设计时应该尽量使代码具有可扩展性和灵活性。
-
例如,可以支持不同大小的棋盘、不同级别的人机对战、网络对战等扩展功能。
5.用户体验:
-
确保游戏界面简洁明了,易于上手。
-
提供合适的提示和反馈,让玩家清晰地了解游戏状态和下一步操作。
6.测试和优化:
-
进行充分的单元测试和集成测试,确保游戏的稳定性和可靠性。
-
根据用户反馈和测试结果进行优化,提升游戏性能和用户体验。
基于上述考虑,可以开始设计和实现Java语言的五子棋游戏。在具体实现过程中,可以借助Java图形界面库(JavaFX)或控制台输入输出来实现用户界面,同时利用Java的面向对象特性来组织代码结构,实现游戏逻辑。
开发步骤:
1.绘制主体框架
使用JavaFX绘制了一个基本框架。只包含了UI的绘制,主要内容包括面板对象、重新开始游戏按钮、悔棋按钮,没有添加具体的按钮功能。
@Overridepublic void start(Stage primaryStage) {BorderPane root = new BorderPane();// 创建棋盘面板对象GridPane boardPane = new GridPane();// 设置棋盘面板样式boardPane.setStyle("-fx-background-color: #FFE4B5; -fx-padding: 10;");// 创建重新开始游戏按钮Button restartButton = new Button("重新开始游戏");// 设置重新开始游戏按钮样式restartButton.setStyle("-fx-font-size: 14;");// 创建悔棋按钮Button undoButton = new Button("悔棋");// 设置悔棋按钮样式undoButton.setStyle("-fx-font-size: 14;");// 创建底部按钮面板BorderPane bottomPane = new BorderPane();bottomPane.setLeft(restartButton);bottomPane.setRight(undoButton);// 设置底部按钮面板样式bottomPane.setStyle("-fx-background-color: #F5F5F5; -fx-padding: 10;");// 将棋盘面板和底部按钮面板添加到主面板中root.setCenter(boardPane);root.setBottom(bottomPane);// 设置主场景Scene scene = new Scene(root, 600, 600);// 设置主舞台标题primaryStage.setTitle("微澜五子棋游戏");// 设置主舞台场景primaryStage.setScene(scene);// 显示主舞台primaryStage.show();}public static void main(String[] args) {launch(args);}
代码效果展示如下:
2.编写按钮事件类
用来监听重新开始游戏按钮和悔棋按钮,并在点击时作出相应的动作。这个类需要在主程序中进行实例化,并设置给对应的按钮。
import javafx.event.ActionEvent;
import javafx.event.EventHandler;
import javafx.scene.control.Button;public class ButtonEventHandler implements EventHandler<ActionEvent> {private GomokuGameUI gameUI; // 引用游戏UI对象public ButtonEventHandler(GomokuGameUI gameUI) {this.gameUI = gameUI;}@Overridepublic void handle(ActionEvent event) {if (event.getSource() instanceof Button) {Button clickedButton = (Button) event.getSource();if (clickedButton.getText().equals("重新开始游戏")) {// 处理重新开始游戏按钮点击事件System.out.println("重新开始游戏按钮被点击了");// 添加重新开始游戏的逻辑} else if (clickedButton.getText().equals("悔棋")) {// 处理悔棋按钮点击事件System.out.println("悔棋按钮被点击了");// 添加悔棋的逻辑}}}
}
创建了一个 ButtonEventHandler
类的实例,并将游戏UI对象传递给它。然后,我们将这个事件处理类与重新开始游戏按钮和悔棋按钮进行关联,以便在点击按钮时执行相应的动作。补充UI对象内容如下:
import javafx.application.Application;
import javafx.scene.Scene;
import javafx.scene.control.Button;
import javafx.scene.layout.BorderPane;
import javafx.scene.layout.GridPane;
import javafx.stage.Stage;public class GomokuGameUI extends Application {@Overridepublic void start(Stage primaryStage) {BorderPane root = new BorderPane();// 其他代码// 创建按钮事件处理类,并将游戏UI对象传入ButtonEventHandler buttonHandler = new ButtonEventHandler(this);// 将按钮事件处理类设置给重新开始游戏按钮和悔棋按钮restartButton.setOnAction(buttonHandler);undoButton.setOnAction(buttonHandler);//其他代码}public static void main(String[] args) {launch(args);}
}
代码效果展示如下:
3.编写棋盘类包含棋子画法
下面是一个简单的棋盘类,用于绘制五子棋的棋盘并管理棋子的放置。在这个棋盘类中,我们将棋盘表示为一个二维数组,用于存储棋子的位置和类型信息。我在这里添加了放置棋子和检查胜利的方法。
import javafx.collections.ObservableList;
import javafx.event.EventHandler;
import javafx.scene.Node;
import javafx.scene.input.MouseEvent;
import javafx.scene.layout.Pane;
import javafx.scene.paint.Color;
import javafx.scene.shape.Circle;
import javafx.scene.shape.Rectangle;public class GomokuBoardPane extends Pane {public static final int BOARD_SIZE = 15;public static final int CELL_SIZE = 40;private char[][] board;//定义玩家信息private char currentPlayer;//在 GomokuBoardPane 类中添加一个变量来跟踪游戏是否结束private boolean gameEnded;public GomokuBoardPane() {board = new char[BOARD_SIZE][BOARD_SIZE];currentPlayer = 'X'; // 初始化为玩家 XinitializeBoard();setOnMouseClicked(new BoardClickHandler());}public void initializeBoard() {//清屏getChildren().clear();for (int row = 0; row < BOARD_SIZE; row++) {for (int col = 0; col < BOARD_SIZE; col++) {board[row][col] = '-';Rectangle cell = new Rectangle(CELL_SIZE, CELL_SIZE);cell.setFill(Color.BISQUE);cell.setStroke(Color.BLACK);cell.setX(20 + col * CELL_SIZE);cell.setY(20 + row * CELL_SIZE);getChildren().add(cell);}}}public void setGameEnded(boolean b) {gameEnded = b;}public void setCurrentPlayer(char x) {currentPlayer = x;}private class BoardClickHandler implements EventHandler<MouseEvent> {@Overridepublic void handle(MouseEvent event) {if (!gameEnded) {int col = (int) (event.getX() / CELL_SIZE);int row = (int) (event.getY() / CELL_SIZE);if (isValidMove(row, col)) {placePiece(currentPlayer, row, col);// 在此处添加其他玩家的逻辑(例如交替落子、检查胜利等)if (checkWin(row, col)) {System.out.println("Player " + currentPlayer + " wins!");// 在此处添加游戏结束的逻辑,例如显示胜利信息、禁用棋盘等// 在 handle 方法中调用 endGame 方法endGame(currentPlayer);} else {currentPlayer = (currentPlayer == 'X') ? 'O' : 'X'; // 切换玩家}}}}}private boolean isValidMove(int row, int col) {return board[row][col] == '-';}private void placePiece(char type, int row, int col) {board[row][col] = type;Circle piece = new Circle(CELL_SIZE / 2, type == 'X' ? Color.BLACK : Color.WHITE);piece.setCenterX(col * CELL_SIZE + CELL_SIZE / 2);piece.setCenterY(row * CELL_SIZE + CELL_SIZE / 2);getChildren().add(piece);}public boolean checkWin(int row, int col) {char piece = board[row][col];// 检查横向int count = 1;for (int i = col - 1; i >= 0 && board[row][i] == piece; i--) {count++;}for (int i = col + 1; i < BOARD_SIZE && board[row][i] == piece; i++) {count++;}if (count >= 5) return true;// 检查纵向count = 1;for (int i = row - 1; i >= 0 && board[i][col] == piece; i--) {count++;}for (int i = row + 1; i < BOARD_SIZE && board[i][col] == piece; i++) {count++;}if (count >= 5) return true;// 检查左上到右下斜向count = 1;for (int i = row - 1, j = col - 1; i >= 0 && j >= 0 && board[i][j] == piece; i--, j--) {count++;}for (int i = row + 1, j = col + 1; i < BOARD_SIZE && j < BOARD_SIZE && board[i][j] == piece; i++, j++) {count++;}if (count >= 5) return true;// 检查右上到左下斜向count = 1;for (int i = row - 1, j = col + 1; i >= 0 && j < BOARD_SIZE && board[i][j] == piece; i--, j++) {count++;}for (int i = row + 1, j = col - 1; i < BOARD_SIZE && j >= 0 && board[i][j] == piece; i++, j--) {count++;}return count >= 5;}// 在游戏结束时调用此方法private void endGame(char winner) {gameEnded = true;System.out.println("Player " + winner + " wins!");// 在这里添加显示胜利信息的逻辑}
}
将主框架里面的初始画棋盘内容替换,ButtonEventHandler 类中的构造方法增加一种。
// 创建棋盘对象GomokuBoardPane boardPane = new GomokuBoardPane();// 将棋盘添加到主面板中root.setCenter(boardPane);// 创建按钮事件处理类,并将游戏UI对象和棋盘对象传入ButtonEventHandler buttonHandler = new ButtonEventHandler(this, boardPane);
代码效果展示如下:
因为判断的基本方法我们已经处理过,到这一步就可以进行游戏。
4.重新开始、悔棋功能
(1)重新开始游戏
机器重启即可,考虑是直接重画一次棋盘最方便。
// 重新开始游戏的逻辑
boardPane.initializeBoard();//重画棋盘
boardPane.setCurrentPlayer('X');//设置玩家
boardPane.setGameEnded(false);//标记游戏状态
(2)悔棋
画棋子的懒省事没有查居中内容。现在计算最后一个棋子的坐标时,需要吧坐标减去20再更新状态。注意切换玩家
// 移除最后一步棋子public void undoMove() {ObservableList<Node> children = getChildren();int lastPieceIndex = children.size() - 1; // 获取最后一个棋子的索引if (lastPieceIndex >= 0) {Node lastPiece = children.get(lastPieceIndex); // 获取最后一个棋子节点if (lastPiece instanceof Circle) {Circle lastCircle = (Circle) lastPiece;int row = (int) ((lastCircle.getCenterY() - 20.0) / CELL_SIZE);int col = (int) ((lastCircle.getCenterX() - 20.0) / CELL_SIZE);// 移除最后一个棋子节点children.remove(lastPiece);// 更新棋盘状态board[row][col] = '-';// 切换当前玩家currentPlayer = (currentPlayer == 'X') ? 'O' : 'X';}}}
5.项目结构
项目到这里就基本完成了。主要实现了玩家对战,重新开始,悔棋等操作,结构简单。适合初学者学习。
结语:
本次开发过程多曲折。本身就不熟悉JavaFx图形库,想着了解一下新内容,结果不知不觉踩了不少坑。棋盘类从GridPane到Pane,也是有一些意想不到事情发生。做事情之前尤其是开发内容应该提前预估一下可能遇到的问题。这次问题好好记录,保证下次不犯类似的问题。(后续有机会继续改进这个项目)
需要完整源码参考的请扫码:
相关文章:

五子棋:不会下五子棋也没关系,会用Java写五子棋就行
关注公号“微澜网络”获取完整源代码! 效果展示: 目录 效果展示: 导语: 游戏介绍: 程序设计: 1.游戏规则和功能: 2.用户界面设计: 3.程序架构设计: 4.可扩展性和灵…...

【VUE】使用Vue和CSS动画创建滚动列表
使用Vue和CSS动画创建滚动列表 在这篇文章中,我们将探讨如何使用Vue.js和CSS动画创建一个动态且视觉上吸引人的滚动列表。这个列表将自动滚动显示项目,类似于轮播图的方式,非常适合用于仪表盘、排行榜或任何需要在有限空间内展示项目列表的应…...

分布式结构化数据表Bigtable
文章目录 设计动机与目标数据模型行列时间戳 系统架构主服务器Chubby作用子表服务器SSTable结构子表实际组成子表地址组成子表数据存储及读/写操作数据压缩 性能优化局部性群组(Locality groups)压缩布隆过滤器 Bigtable是Google开发的基于GFS和Chubby的…...
langchain 加载 csv,json
csv from langchain_community.document_loaders.csv_loader import CSVLoaderloader CSVLoader(file_pathdata/专业描述.csv, csv_args{delimiter: ,,quotechar: ",fieldnames: [专业, 描述] }, encodingutf8, source_column专业)data loader.load() print(data)quote…...
Java-常见面试题收集(十三)
二十二 Redis 1 Redis 作用 Redis,全称Remote Dictionary Server,即远程字典服务,是一个开源的使用ANSI C语言编写的、支持网络的、基于内存亦可持久化的日志型、Key-Value数据库,并提供多种语言的API。它主要用于缓存数据的计算…...

第二证券策略:股指预计维持震荡格局 关注汽车、工程机械等板块
第二证券指出,指数自今年2月份阶段低点反弹以来,3月份持续高位整理。进入4月份之后面对年报和一季报的双重财报发表期,预计指数短期保持高位整理概率比较大。前期缺乏成绩支撑的概念股或有回落的危险,主张重视成绩稳定、估值低、分…...

hcia datacom课程学习(6):路由与路由表基础
1.路由的作用 不同网段的设备互相通信需要具有路由功能的设备进行转发 具有路由功能的设备不一定是路由器,交换机可以有路由功能,同样的,路由器也可以有交换功能,像家里常用的路由器就是集路由功能和交换功能于一体的 2.路由相…...

AI PC元年,华为的一张航海图、一艘渡轮和一张船票
今天,从学术研究者到产业投资者,无不认为大模型掀起了一场人工智能的完美风暴。 所谓“完美风暴”,指的是一项新技术的各个要素,以新的方式互相影响、彼此加强,组合在一起形成了摧枯拉朽般的力量。 而我们每个人&#…...

NAT技术
网络技术深似海呀,一段时间不用又忘。 是什么 NAT技术是网络防火墙技术的一部分,可以作用在linux防火墙或者设备防火墙,NAT技术可以实现地址和端口的转换,主要还是为了网络连通性。 作用 存在以下三个IP,A(10.234.…...

新能源汽车“价格战”之后,充电桩主板市场将会怎样?
2024年2月底,国内新能源汽车市场开启了一场前所未有的“价格战”↓ 比亚迪率先抛出“王炸”车型——秦PLUS荣耀版和驱逐舰05荣耀版,起售价低至7.98万元,打响了价格战的“第一枪”,引爆了平静的汽车市场。 “电比油低”就此拉开序…...

appium driver install uiautomator2 安装失败
报错 Installing ‘uiautomator2’ using NPM install spec ‘appium-uiautomator2-driver’ Error: Encountered an error when installing package: npm command ‘install --save-dev --no-progress --no-audit --omitpeer --save-exact --global-style --no-package-lock…...

学浪已购买视频怎么下载到本地?
许多学习者在学浪购买了丰富的课程,然而,一些课程存在时间限制,使得学习者希望将其下载并永久保存。在这里,我们将介绍一款名为小浪助手的工具,它能够帮助你轻松将学浪已购买的视频下载到本地,让学习变得更…...
k8s-pod设置执行优先级
Pod的优先级管理是Kubernetes调度中的一个重要特性,通过PriorityClass(优先级类)的设置,我们可以为Pod指定不同的优先级,从而在资源有限的情况下更精细地调整调度顺序 什么是PriorityClass? PriorityClass是…...
const修饰指针
const修饰指针 常量指针 特点为指针的指向可以改,但是指针指向的值不可以修改 int a 10; int b 20; const int *p &a; *p 20; //错误,指针的指向的值不可更改 p &b; //正确 指针常量 特点是指针的指向不可以改,指针指向的值…...
php关于序列化r的指向
在PHP中,序列化字符串的索引是根据序列化过程中值的出现顺序来确定的。每个值(包括数组的键和值)在序列化字符串中都会被赋予一个顺序索引。为了理解这个顺序,我们需要知道以下几点: 序列化时,数组的键和值…...

从0到1实现RPC | 11 丰富测试案例
测试案例主要针对服务消费者consumer,复杂逻辑都在consumer端。 常规int类型,返回User对象 参数类型转换,主要实现逻辑都在TypeUtils工具类中。 测试方法重载,同名方法,参数不同 方法签名的实现,主要逻辑…...
在前端开发中用到了哪些设计模式?
在前端开发中用到了哪些设计模式? 1.单例模式2.观察者模式3.工厂模式4.适配器模式5.装饰器模式6.命令模式7.迭代器模式8.组合模式9.策略模式10.发布订阅模式 1.单例模式 确保一个类只有一个实例,提供一个全局访问点,vue就是一个单例模式&…...
ES6 的解构赋值
解构赋值(Destructuring assignment)是一种方便快捷的方式,可以从对象或数组中提取数据,并将数据赋值给变量。解构赋值是ES6中一项强大且常用的特性. 1. 基本数组解构 首先,让我们看看如何对数组进行解构赋值。假设我…...

蓝桥杯物联网竞赛_STM32L071KBU6_全部工程及国赛省赛真题及代码
包含stm32L071kbu6全部实验工程、源码、原理图、官方提供参考代码及国、省赛真题及代码 链接:https://pan.baidu.com/s/1pXnsMHE0t4RLCeluFhFpAg?pwdq497 提取码:q497...

关于UCG游戏平台的一些思考
UCG游戏平台,全称User Generated Content,即用户生成内容。它涵盖了所有玩家可以自主编辑的部分,包含并不限于换装、捏脸、关卡摆放等内容。 UCG概念在最近又火了起来,但这个模式出现的并不早。早在10多年前,war3编辑器…...

shell脚本--常见案例
1、自动备份文件或目录 2、批量重命名文件 3、查找并删除指定名称的文件: 4、批量删除文件 5、查找并替换文件内容 6、批量创建文件 7、创建文件夹并移动文件 8、在文件夹中查找文件...
JAVA后端开发——多租户
数据隔离是多租户系统中的核心概念,确保一个租户(在这个系统中可能是一个公司或一个独立的客户)的数据对其他租户是不可见的。在 RuoYi 框架(您当前项目所使用的基础框架)中,这通常是通过在数据表中增加一个…...

day36-多路IO复用
一、基本概念 (服务器多客户端模型) 定义:单线程或单进程同时监测若干个文件描述符是否可以执行IO操作的能力 作用:应用程序通常需要处理来自多条事件流中的事件,比如我现在用的电脑,需要同时处理键盘鼠标…...

解析奥地利 XARION激光超声检测系统:无膜光学麦克风 + 无耦合剂的技术协同优势及多元应用
在工业制造领域,无损检测(NDT)的精度与效率直接影响产品质量与生产安全。奥地利 XARION开发的激光超声精密检测系统,以非接触式光学麦克风技术为核心,打破传统检测瓶颈,为半导体、航空航天、汽车制造等行业提供了高灵敏…...

Vue ③-生命周期 || 脚手架
生命周期 思考:什么时候可以发送初始化渲染请求?(越早越好) 什么时候可以开始操作dom?(至少dom得渲染出来) Vue生命周期: 一个Vue实例从 创建 到 销毁 的整个过程。 生命周期四个…...

Cilium动手实验室: 精通之旅---13.Cilium LoadBalancer IPAM and L2 Service Announcement
Cilium动手实验室: 精通之旅---13.Cilium LoadBalancer IPAM and L2 Service Announcement 1. LAB环境2. L2公告策略2.1 部署Death Star2.2 访问服务2.3 部署L2公告策略2.4 服务宣告 3. 可视化 ARP 流量3.1 部署新服务3.2 准备可视化3.3 再次请求 4. 自动IPAM4.1 IPAM Pool4.2 …...

Sklearn 机器学习 缺失值处理 获取填充失值的统计值
💖亲爱的技术爱好者们,热烈欢迎来到 Kant2048 的博客!我是 Thomas Kant,很开心能在CSDN上与你们相遇~💖 本博客的精华专栏: 【自动化测试】 【测试经验】 【人工智能】 【Python】 使用 Scikit-learn 处理缺失值并提取填充统计信息的完整指南 在机器学习项目中,数据清…...
深度解析:etcd 在 Milvus 向量数据库中的关键作用
目录 🚀 深度解析:etcd 在 Milvus 向量数据库中的关键作用 💡 什么是 etcd? 🧠 Milvus 架构简介 📦 etcd 在 Milvus 中的核心作用 🔧 实际工作流程示意 ⚠️ 如果 etcd 出现问题会怎样&am…...

可视化预警系统:如何实现生产风险的实时监控?
在生产环境中,风险无处不在,而传统的监控方式往往只能事后补救,难以做到提前预警。但如今,可视化预警系统正在改变这一切!它能够实时收集和分析生产数据,通过直观的图表和警报,让管理者第一时间…...
win11部署suna
参考链接 项目链接 沙盒链接 数据库链接 本文介绍 本文只为项目的辅助,手把手太麻烦 执行步骤 1.下载代码 git clone https://github.com/kortix-ai/suna.git cd suna2.配置环境(在Anaconda Prompt上执行) python setup.py3.运行代码 …...