五子棋(C语言实现)
目录
构思
1、主程序
2、初始化
3、游戏菜单
4、打印棋盘
6、玩家下棋
7、判断输赢
8、功能整合
人机下棋
完整版:
game.h
game.c
text.c
测试功能代码
构思
五子棋不必多介绍了,大家小时候都玩过哈。
我们要通过程序实现这个小游戏,大体上的构思得有:
游戏流程:运行游戏>>打印棋盘>>下棋>>判断输赢
由此,我们声明如下函数、棋盘数组和回合数 (用于双人下棋时下不同棋子的判断)。
将这些都放在我创建的头文件game.h中方便其他文件使用:
#include <stdio.h>
#include <Windows.h>int map[19][19];//棋盘
int flag;//回合数//初始化棋盘
void init();//游戏菜单
void menuView();//打印棋盘
void gameView_ShowMap();//玩家下棋
int playerMove(int x, int y);//判断输赢
int isWin(int x, int y);//判断输赢
void winView();
1、主程序
先进行游戏下棋,后进行判断输赢,比较符合do—while的特点。
int main()
{int input = 0;do{menuView();printf("请选择:");scanf("%d", &input);switch (input) {case 1:gameView();break;case 0:printf("退出游戏\n");break;default:printf("输入错误,请重新选择.\n");break;}} while (input);return 0;
}
- 首先打印菜单,提示玩家进行选择开始游戏。
- 通过输入变量input的值,判断是否进行游戏,
- 值为1则调用gameView函数进行游戏,0则结束游戏。
- gameView函数对各种功能函数进行整合。
在此之前先将各种功能函数进行实现,最后将它们放入gameView整合。
2、初始化
void init() {for (int i = 0; i < 19; i++) {for (int j = 0; j < 19; j++) {map[i][j] = 0;}}flag = 0;
}
- 将棋盘每个格子初始化为0。
- 回合数初始化为0。
3、游戏菜单
void menuView() {printf("*************************\n");printf("***** 1. play ******\n");printf("***** 0. exit ******\n");printf("*************************\n");
}
4、打印棋盘
这里我选择设计了棋盘格和横纵坐标,我们可以根据需要自行设计。
void gameView_ShowMap() {int i, j;printf(" ");for (i = 0; i < 19; i++) {//打印横坐标printf("%3d ",i);}printf("\n ");for (i = 0; i < 19; i++) {printf("+---");}printf("+\n");for (i = 0; i < 19; i++) {printf("%2d |", i);//每行输出前都先打印纵坐标for (j = 0; j < 19; j++) {printf(" %d ", map[i][j]);//打印棋子if (j < 18)printf("|");}printf("|\n");if(i<18)printf(" |");if(i==18)printf(" +");for (j = 0; j < 19; j++) {printf("---");if (j < 18)printf("+");}printf("+\n");}
}
效果如下:
6、玩家下棋
int playerMove(int x, int y) {if (x >= 0 && x < 19 && y >= 0 && y < 19) {if (map[x][y] == 0) {if (flag % 2 == 0)map[x][y] = 1;//下黑子elsemap[x][y] = 2;//下白子//落子成功return 1;}else {//该位置已有棋子return 0;}}else {//坐标不合法return -1;}
}
- 首先判断下棋位置是否在棋盘0到18的横纵坐标内,如果不在函数返回值-1,
- 如果在棋盘内,则进行下棋。
- 棋盘每个格子初始值为0,判断当前格子的值为0才可以下棋,否则返回值为0。
- 回合数flag初始值为0,当flag为偶数下黑子>>1,奇数下白子>>2,成功下棋返回值为1。
7、判断输赢
int isWin(int x, int y) {int i, j;for (i = 0; i < 19; i++) {for (j = 0; j < 19; j++) {if (map[i][j] == 0) {continue;}//横着连成五子if (j < 15)if (map[i][j] == map[i][j + 1] && map[i][j] == map[i][j + 2]&& map[i][j] == map[i][j + 3] && map[i][j] == map[i][j + 4])return map[i][j];//竖着连成五子if (i < 15)if (map[i][j] == map[i + 1][j] && map[i][j] == map[i + 2][j]&& map[i][j] == map[i + 3][j] && map[i][j] == map[i + 4][j])return map[i][j];//左斜着连成五子-> " \ "if (i < 15 && j < 15)if (map[i][j] == map[i + 1][j + 1] && map[i][j] == map[i + 2][j + 2]&& map[i][j] == map[i + 3][j + 3] && map[i][j] == map[i + 4][j + 4])return map[i][j];//右斜着连成五子-> " / "if (i < 15 && j > 4)if (map[i][j] == map[i + 1][j - 1] && map[i][j] == map[i + 2][j - 2]&& map[i][j] == map[i + 3][j - 3] && map[i][j] == map[i + 4][j - 4])return map[i][j];}}return 0;
}
-
i < 15
用于确保在检查垂直和左斜方向的连续五子时,起始位置(i, j)
之后至少还有4个位置。因为五子连成一线需要五个连续的位置,所以确保从当前位置开始往下检查的时候不会超出数组的边界。 -
j < 15
用于确保在检查水平和左斜方向的连续五子时,起始位置(i, j)
之后至少还有4个位置。同样,这是为了确保从当前位置开始往右检查的时候不会超出数组的边界。 -
j > 4
用于确保在检查右斜方向的连续五子时,起始位置(i, j)
之前至少还有4个位置。这是为了确保从当前位置开始往左检查的时候不会超出数组的边界。
8、功能整合
void gameView()
{init();int x = 0, y = 0;int ret = 0;//辅助判断坐标是否合法while (1) {gameView_ShowMap();printf("请输入要下棋的坐标:");scanf("%d %d", &x, &y);if (playerMove(x, y) == 0) {printf("\n!!!该坐标已被占用!!!\n");Sleep(2000);continue;}else if (playerMove(x, y) == -1) {printf("\n!!!请输入合法坐标!!!\n");Sleep(2000);continue;}else {flag++;//切换回合}if (isWin(x, y) == 0) {continue;}else if (isWin(x, y) == 1) {printf("黑子获胜\n");break;}else if (isWin(x, y) == 2) {printf("白子获胜\n");break;}}}
- 这里的Sleep函数需要头文件#include <Windows.h>,使用该函数暂停两秒,防止continue后下次循环打印的棋盘将提示信息挡住。
- 下棋坐标不合法打印提示信息后,进入下次循环重新输入。
- 成功下棋则flag自增,切换回合。
- 每次下棋后都要判断输赢,有人赢了则停止循环,否则继续下棋。
- (其实应该从下棋次数第五次开始判断输赢更合理,读者可以自行添加判断)
人机下棋
想要实现人机下棋可以看看我这篇文章《三子棋》 ,里面实现了人机下棋,读者可以自行模仿改进,实现其功能的重要函数可以在这篇文章中学习 rand&srand函数 。
完整版:
game.h
#include <stdio.h>
#include <Windows.h>
int map[19][19];//棋盘
int flag;//回合数//初始化
void init();//游戏菜单
void menuView();//打印棋盘
void gameView_ShowMap();//玩家下棋
int playerMove(int x, int y);//判断输赢
int isWin(int x, int y);
game.c
#define _CRT_SECURE_NO_WARNINGS 1
#include "game.h"void init() {for (int i = 0; i < 19; i++) {for (int j = 0; j < 19; j++) {map[i][j] = 0;}}flag = 1;
}int isWin(int x, int y) {int i, j;for (i = 0; i < 19; i++) {for (j = 0; j < 19; j++) {if (map[i][j] == 0) {continue;}//横着连成五子if (j < 15)if (map[i][j] == map[i][j + 1] && map[i][j] == map[i][j + 2]&& map[i][j] == map[i][j + 3] && map[i][j] == map[i][j + 4])return map[i][j];//竖着连成五子if (i < 15)if (map[i][j] == map[i + 1][j] && map[i][j] == map[i + 2][j]&& map[i][j] == map[i + 3][j] && map[i][j] == map[i + 4][j])return map[i][j];//左斜着连成五子-> " \ "if (i < 15 && j < 15)if (map[i][j] == map[i + 1][j + 1] && map[i][j] == map[i + 2][j + 2]&& map[i][j] == map[i + 3][j + 3] && map[i][j] == map[i + 4][j + 4])return map[i][j];//右斜着连成五子-> " / "if (i < 15 && j > 4)if (map[i][j] == map[i + 1][j - 1] && map[i][j] == map[i + 2][j - 2]&& map[i][j] == map[i + 3][j - 3] && map[i][j] == map[i + 4][j - 4])return map[i][j];}}return 0;
}int playerMove(int x, int y) {if (x >= 0 && x < 19 && y >= 0 && y < 19) {if (map[x][y] == 0) {if (flag % 2 == 0)map[x][y] = 1;//下黑子elsemap[x][y] = 2;//下白子//落子成功return 1;}else {//该位置已有棋子return 0;}}else {//坐标不合法return -1;}
}void menuView() {printf("*************************\n");printf("***** 1. play ******\n");printf("***** 0. exit ******\n");printf("*************************\n");
}void gameView_ShowMap() {int i, j;printf(" ");for (i = 0; i < 19; i++) {//打印横坐标printf("%3d ",i);}printf("\n ");for (i = 0; i < 19; i++) {printf("+---");}printf("+\n");for (i = 0; i < 19; i++) {printf("%2d |", i);//每行输出前先都打印纵坐标for (j = 0; j < 19; j++) {printf(" %d ", map[i][j]);if (j < 18)printf("|");}printf("|\n");if(i<18)printf(" |");if(i==18)printf(" +");for (j = 0; j < 19; j++) {printf("---");if (j < 18)printf("+");}printf("+\n");}
}
text.c
#define _CRT_SECURE_NO_WARNINGS 1
#include "game.h"void gameView()
{init();int x = 0, y = 0;int ret = 0;//辅助判断坐标是否合法while (1) {gameView_ShowMap();printf("请输入要下棋的坐标:");scanf("%d %d", &x, &y);if (playerMove(x, y) == 0) {printf("\n!!!该坐标已被占用!!!\n");Sleep(2000);continue;}else if (playerMove(x, y) == -1) {printf("\n!!!请输入合法坐标!!!\n");Sleep(2000);continue;}else {flag++;//切换回合}if (isWin(x, y) == 0) {continue;}else if (isWin(x, y) == 1) {printf("黑子获胜\n");break;}else if (isWin(x, y) == 2) {printf("白子获胜\n");break;}}}int main()
{int input = 0;do{menuView();printf("请选择:");scanf("%d", &input);switch (input) {case 1:gameView();break;case 0:printf("退出游戏\n");break;default:printf("输入错误,请重新选择.\n");break;}} while (input);return 0;
}
测试功能代码
此代码可以替换int main主函数,用于测试函数功能是否正确,可以减少自行下棋测试时间
成功运行输出如下:
int main()
{int testflag = 0;//init测试代码init();if (flag != 0) {printf("init()错误");exit(0);}for (int i = 0; i < 19; i++) {for (int j = 0; j < 19; j++) {if (map[i][j]) {printf("init()错误");exit(0);}}}printf("init()测试成功\n");testflag++;//playerMove测试代码int result = 1;result &= playerMove(2, 2);result &= playerMove(2, 3);result &= playerMove(2, 4);result &= playerMove(2, 5);if (result != 1 || (map[2][2] && map[2][3] && map[2][4] && map[2][5]) != 1) {printf("playerMove()错误");exit(0);}flag = 1;result &= playerMove(2, 5);if (result != 0 || map[2][5] != 1) {printf("playerMove()错误");exit(0);}printf("playerMove()测试成功\n");testflag++;//isWin测试代码playerMove(2, 1);if (isWin(2, 1)) {printf("isWin()错误");exit(0);}playerMove(1, 0);playerMove(3, 2);playerMove(4, 3);playerMove(5, 4);if (isWin(1, 0) != 2) {printf("isWin()错误");exit(0);}printf("isWin()测试成功\n");testflag++;if (testflag == 3) {printf("service代码测试成功\n");}return 0;
}
相关文章:

五子棋(C语言实现)
目录 构思 1、主程序 2、初始化 3、游戏菜单 4、打印棋盘 6、玩家下棋 7、判断输赢 8、功能整合 人机下棋 完整版: game.h game.c text.c 测试功能代码 构思 五子棋不必多介绍了,大家小时候都玩过哈。 我们要通过程序实现这个小游戏&…...
thymeleaf,bootstrap-fileinput 多文件上传
组件遍历上传 一、前端 <!DOCTYPE html> <html lang"zh" xmlns:th"http://www.thymeleaf.org" > <head><th:block th:include"include :: header(修改固定资产信息)" /><th:block th:include"include :: date…...

爬虫 | 基础模块了解
文章目录 📚http协议📚requests模块📚re模块🐇 re.I 或 re.IGNORECASE🐇re.M或 re.MULTILINE🐇re.S 或 re.DOTALL🐇 re.A 或 re.ASCII🐇 re.X 或 re.VERBOSE🐇特殊字符类…...

CSS复习笔记
CSS 文章目录 CSS1.概念2.CSS 引入方式3.选择器基础选择器:标签选择器类选择器id 选择器通配符选择器 复合选择器:**后代选择器****子代选择器****并集选择器****交集选择器-了解****伪类选择器** 结构伪类选择器:**:nth-child(公式)**伪元素…...

编译linux的设备树
使用make dtbs命令时 在arch/arm 的目录Makefile文件中有 boot : arch/arm/boot prepare 和scripts是空的 在文件scripts/Kbuild.include中 变量build : -f $(srctree)/scripts/Makefile.build obj build变量虽然没有在arch/arm 的目录Makefile文件中定义,但…...
⛳ MyBatis 中 Mapper 接口工作原理实例解析
🎍目录 ⛳ MyBatis 中 Mapper 接口工作原理实例解析🎨 一、Mapper 接口是怎么找到实现类的?🐾 二、从一段代码看起🚜 三、Mapper 接口🏭 四、Mapper 接口的动态代理类的生成🎁 五、总结 ⛳ MyBa…...

Android 音频可视化
Android音频可视化,指的是将音频的频率绘制到屏幕上,达到一种视觉效果,使播放或录制过程更加生动形象。 在Android进行视频可视化涉及的三个主要知识点,其中比较难以理解的傅里叶变换公式。 Android原生的Visualizer使用(获取频…...
刷机与救砖避坑指南
提示:快速进行刷机和救砖学习理解 文章目录 一、刷机1.什么是刷机,需要进行那些准备?2.刷机1.解开bl(bootloader)锁2.刷入TWRP和Magsik3.刷入第三方ROM 二、救砖(9008)1.手机售后一键线刷包&…...
软件建模知识点
提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 前言一、pandas是什么?二、使用步骤 1.引入库2.读入数据总结 前言 提示:这里可以添加本文要记录的大概内容: 例如:…...

WSL 配置 Linux
WSL 配置 Linux Windows 启动 Linux 子系统 控制面板 -> 程序和功能, 将 适用于 Linux 的 Windows 子系统 勾选。 安装 Terminal 在 Microsoft Store 市场上搜索 Terminal 安装 Windows Terminal。 安装 编译工具链 sudo apt update # 更新软件包 sudo apt i…...

VS Code:CMake配置
概述 在VSCode和编译器MinGW安装完毕后,要更高效率的进行C/C开发,采用CMake。CMake是一个开源、跨平台的编译、测试和打包工具,它使用比较简单的语言描述编译,安装的过程,输出Makefile或者project文件,再去…...

Flex 词法分析实验实现(电子科技大学编译技术Icoding实验)
Flex 词法分析 此为电子科技大学编译技术 实验1:词法分析 将具体实现中的三个文件和自己的实验报告一起上传才能通过 根据词法分析实验中给定的文法,利用 flex 设计一词法分析器,该分析器从标准输入读入源代码后,输出单词的类别编…...
设计模式——20. 解释器模式
1. 说明 解释器模式(Interpreter Pattern)是一种行为型设计模式,它用于定义一门语言的语法解析,并为该语言创建解释器。该模式将一个问题或领域表达成一个语言,然后提供一个解释器来解释这种语言中的表达式,以执行特定操作。 要点和组成部分: 抽象表达式(Abstract Ex…...

多输入多输出 | MATLAB实现CNN-BiLSTM-Attention卷积神经网络-双向长短期记忆网络结合SE注意力机制的多输入多输出预测
MATLAB实现CNN-BiLSTM-Attention卷积神经网络-双向长短期记忆网络结合SE注意力机制的多输入多输出预测 目录 MATLAB实现CNN-BiLSTM-Attention卷积神经网络-双向长短期记忆网络结合SE注意力机制的多输入多输出预测预测效果基本介绍程序设计往期精彩参考资料 预测效果 基本介绍 C…...

一文让你玩转Linux多进程开发
Linux多进程开发 主要介绍多进程开发时的要点 进程状态转换 进程反应了进程执行的变化。 进程的状态分为三种 ,运行态,阻塞态,就绪态 在五态模型中分为以下几种,新建态,就绪态,运行态,阻塞态,终止态。 运行态:进程占用处理器正在运…...

Linux线程同步实例
线程同步实例 1. 生产消费者模型基本概念2. 基于BlockingQueue的生产者消费者模型3. 基于环形队列的生产消费模型4. 线程池 1. 生产消费者模型基本概念 生产者消费者模型是一种常用的并发设计模式,它可以解决生产者和消费者之间的速度不匹配、解耦、异步等问题。生…...
LuatOS-SOC接口文档(air780E)-- iconv - iconv操作
iconv.open(tocode, fromcode)# 打开相应字符编码转换函数 参数 传入值类型 解释 string 释义:目标编码格式 取值:gb2312/ucs2/ucs2be/utf8 string 释义:源编码格式 取值:gb2312/ucs2/ucs2be/utf8 返回值 返回值类型 解…...

matlab第三方硬件支持包下载和安装
1、在使用matlab内部的附加功能安装时,由于matlab会验证是否正版无法打开 2、在matlab官网直接找到对应的硬件支持包下载,但是是下图的安装程序 可以直接在matlab中跳转到该程序所在的文件夹双击安装,但是安装到最后出错了 3.根据出错时mala…...

docker compose和consul(服务注册与发现)
一、Docker-compose 简介 Docker-Compose项目是基于Python开发的Docker官方开源项目,负责实现对Docker容器集群的快速编排。 Docker-Compose将所管理的容器分为三层,分别是 工程(project),服务(service&a…...

使用Python进行钻石价格分析
钻石是最昂贵的宝石之一。钻石的质量通常以其重量(克拉)、净度、颜色和切工来评估。重量越大、净度越高、色彩纯净、切工精细的钻石价格也越高。其中,4C标准是衡量钻石质量的国际标准,即克拉(Carat)、净度&…...

【kafka】Golang实现分布式Masscan任务调度系统
要求: 输出两个程序,一个命令行程序(命令行参数用flag)和一个服务端程序。 命令行程序支持通过命令行参数配置下发IP或IP段、端口、扫描带宽,然后将消息推送到kafka里面。 服务端程序: 从kafka消费者接收…...
应用升级/灾备测试时使用guarantee 闪回点迅速回退
1.场景 应用要升级,当升级失败时,数据库回退到升级前. 要测试系统,测试完成后,数据库要回退到测试前。 相对于RMAN恢复需要很长时间, 数据库闪回只需要几分钟。 2.技术实现 数据库设置 2个db_recovery参数 创建guarantee闪回点,不需要开启数据库闪回。…...

练习(含atoi的模拟实现,自定义类型等练习)
一、结构体大小的计算及位段 (结构体大小计算及位段 详解请看:自定义类型:结构体进阶-CSDN博客) 1.在32位系统环境,编译选项为4字节对齐,那么sizeof(A)和sizeof(B)是多少? #pragma pack(4)st…...

Docker 运行 Kafka 带 SASL 认证教程
Docker 运行 Kafka 带 SASL 认证教程 Docker 运行 Kafka 带 SASL 认证教程一、说明二、环境准备三、编写 Docker Compose 和 jaas文件docker-compose.yml代码说明:server_jaas.conf 四、启动服务五、验证服务六、连接kafka服务七、总结 Docker 运行 Kafka 带 SASL 认…...

关于nvm与node.js
1 安装nvm 安装过程中手动修改 nvm的安装路径, 以及修改 通过nvm安装node后正在使用的node的存放目录【这句话可能难以理解,但接着往下看你就了然了】 2 修改nvm中settings.txt文件配置 nvm安装成功后,通常在该文件中会出现以下配置&…...

UE5 学习系列(三)创建和移动物体
这篇博客是该系列的第三篇,是在之前两篇博客的基础上展开,主要介绍如何在操作界面中创建和拖动物体,这篇博客跟随的视频链接如下: B 站视频:s03-创建和移动物体 如果你不打算开之前的博客并且对UE5 比较熟的话按照以…...

Cilium动手实验室: 精通之旅---20.Isovalent Enterprise for Cilium: Zero Trust Visibility
Cilium动手实验室: 精通之旅---20.Isovalent Enterprise for Cilium: Zero Trust Visibility 1. 实验室环境1.1 实验室环境1.2 小测试 2. The Endor System2.1 部署应用2.2 检查现有策略 3. Cilium 策略实体3.1 创建 allow-all 网络策略3.2 在 Hubble CLI 中验证网络策略源3.3 …...

SpringBoot+uniapp 的 Champion 俱乐部微信小程序设计与实现,论文初版实现
摘要 本论文旨在设计并实现基于 SpringBoot 和 uniapp 的 Champion 俱乐部微信小程序,以满足俱乐部线上活动推广、会员管理、社交互动等需求。通过 SpringBoot 搭建后端服务,提供稳定高效的数据处理与业务逻辑支持;利用 uniapp 实现跨平台前…...

现代密码学 | 椭圆曲线密码学—附py代码
Elliptic Curve Cryptography 椭圆曲线密码学(ECC)是一种基于有限域上椭圆曲线数学特性的公钥加密技术。其核心原理涉及椭圆曲线的代数性质、离散对数问题以及有限域上的运算。 椭圆曲线密码学是多种数字签名算法的基础,例如椭圆曲线数字签…...

自然语言处理——Transformer
自然语言处理——Transformer 自注意力机制多头注意力机制Transformer 虽然循环神经网络可以对具有序列特性的数据非常有效,它能挖掘数据中的时序信息以及语义信息,但是它有一个很大的缺陷——很难并行化。 我们可以考虑用CNN来替代RNN,但是…...