五子棋(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)、净度&…...
RestClient
什么是RestClient RestClient 是 Elasticsearch 官方提供的 Java 低级 REST 客户端,它允许HTTP与Elasticsearch 集群通信,而无需处理 JSON 序列化/反序列化等底层细节。它是 Elasticsearch Java API 客户端的基础。 RestClient 主要特点 轻量级ÿ…...
idea大量爆红问题解决
问题描述 在学习和工作中,idea是程序员不可缺少的一个工具,但是突然在有些时候就会出现大量爆红的问题,发现无法跳转,无论是关机重启或者是替换root都无法解决 就是如上所展示的问题,但是程序依然可以启动。 问题解决…...
基于FPGA的PID算法学习———实现PID比例控制算法
基于FPGA的PID算法学习 前言一、PID算法分析二、PID仿真分析1. PID代码2.PI代码3.P代码4.顶层5.测试文件6.仿真波形 总结 前言 学习内容:参考网站: PID算法控制 PID即:Proportional(比例)、Integral(积分&…...
边缘计算医疗风险自查APP开发方案
核心目标:在便携设备(智能手表/家用检测仪)部署轻量化疾病预测模型,实现低延迟、隐私安全的实时健康风险评估。 一、技术架构设计 #mermaid-svg-iuNaeeLK2YoFKfao {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg…...
汽车生产虚拟实训中的技能提升与生产优化
在制造业蓬勃发展的大背景下,虚拟教学实训宛如一颗璀璨的新星,正发挥着不可或缺且日益凸显的关键作用,源源不断地为企业的稳健前行与创新发展注入磅礴强大的动力。就以汽车制造企业这一极具代表性的行业主体为例,汽车生产线上各类…...
全球首个30米分辨率湿地数据集(2000—2022)
数据简介 今天我们分享的数据是全球30米分辨率湿地数据集,包含8种湿地亚类,该数据以0.5X0.5的瓦片存储,我们整理了所有属于中国的瓦片名称与其对应省份,方便大家研究使用。 该数据集作为全球首个30米分辨率、覆盖2000–2022年时间…...
macOS多出来了:Google云端硬盘、YouTube、表格、幻灯片、Gmail、Google文档等应用
文章目录 问题现象问题原因解决办法 问题现象 macOS启动台(Launchpad)多出来了:Google云端硬盘、YouTube、表格、幻灯片、Gmail、Google文档等应用。 问题原因 很明显,都是Google家的办公全家桶。这些应用并不是通过独立安装的…...
高等数学(下)题型笔记(八)空间解析几何与向量代数
目录 0 前言 1 向量的点乘 1.1 基本公式 1.2 例题 2 向量的叉乘 2.1 基础知识 2.2 例题 3 空间平面方程 3.1 基础知识 3.2 例题 4 空间直线方程 4.1 基础知识 4.2 例题 5 旋转曲面及其方程 5.1 基础知识 5.2 例题 6 空间曲面的法线与切平面 6.1 基础知识 6.2…...
Caliper 负载(Workload)详细解析
Caliper 负载(Workload)详细解析 负载(Workload)是 Caliper 性能测试的核心部分,它定义了测试期间要执行的具体合约调用行为和交易模式。下面我将全面深入地讲解负载的各个方面。 一、负载模块基本结构 一个典型的负载模块(如 workload.js)包含以下基本结构: use strict;/…...
day36-多路IO复用
一、基本概念 (服务器多客户端模型) 定义:单线程或单进程同时监测若干个文件描述符是否可以执行IO操作的能力 作用:应用程序通常需要处理来自多条事件流中的事件,比如我现在用的电脑,需要同时处理键盘鼠标…...
