C语言实现简易n子棋小游戏(代码含注解)

利用C语言简单实现一个n子棋小游戏,棋盘大小由自己定义

将源文件分为   执行游戏的测试文件(test.c)和保存游戏运行逻辑的相关函数的文件(game.c)

头文件中声明符号和函数的定义(game.h)

游戏执行主要依靠二维数组实现,电脑走棋采用随机值的方法简易地实现而非采用高级的算法

头文件 game.h

#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <string.h>
#include <math.h>
#include<stdlib.h>
#include<windows.h>
#include<time.h>
//符号的定义(定义几子棋)
#define ROW 3
#define COL 3//函数的定义
void InitBoard(char board[ROW][COL], int row, int col);//初始化棋盘
void PrintBoard(char board[ROW][COL], int row, int col);//打印棋盘
void PlayerMove(char board[ROW][COL], int row, int col);//玩家下棋
void ComputerMove(char board[ROW][COL], int row, int col);//电脑下棋
char IsWin(char board[ROW][COL], int row, int col);//判断游戏结果

源文件 test.c

#include "game.h"            //引用头文件
void game() {char ret;char board[ROW][COL];  //二维数组存储游戏数据,ROW、COL为行和列,在头文件game.h中定义InitBoard(board,ROW,COL);//初始化棋盘为空格PrintBoard(board, ROW, COL);//打印棋盘while (1) {printf("玩家走\n\n");PlayerMove(board, ROW, COL);  //玩家下棋Sleep(500);PrintBoard(board, ROW, COL);  //判断玩家是否赢得游戏ret=IsWin(board, ROW, COL);		//返回值为*代表玩家获胜、#代表电脑获胜、D代表平局、C代表未出结果,游戏继续if (ret != 'C') {break;}printf("电脑走\n\n");ComputerMove(board, ROW, COL);//玩家下棋Sleep(1300);PrintBoard(board, ROW, COL);//判断电脑是否赢得游戏ret = IsWin(board, ROW, COL);if (ret != 'C') {break;}}if (ret == '*') {printf("玩家获胜\n\n");}else if (ret =='#') {printf("电脑获胜\n\n");}else if (ret == 'D') {printf("平局\n\n");}PrintBoard(board, ROW, COL);//打印游戏结果}
void menu() {printf("***************************\n");printf("*****      1.play     *****\n");printf("*****      0.exit     *****\n");printf("***************************\n");}
int main() {int n;srand((unsigned int)time(NULL));//设置随机数种子do{menu();printf("请选择:\n");scanf("%d", &n);switch (n){case 1:game();break;case 0:printf("退出游戏\n");break;default:printf("选择错误,重新选择\n");break;}} while (n);return 0;
}

源文件 game.c

#include "game.h"                        //引用头文件
void InitBoard(char board[ROW][COL], int row, int col) {for (int i = 0; i < ROW; i++) {for (int j = 0; j < COL; j++) {board[i][j] = ' ';}}
}
void PrintBoard(char board[ROW][COL], int row, int col) {          //打印棋盘for (int j = 0; j < col; j++) {printf(" ___");}printf(" \n");for (int i = 0; i < row; i++) {for (int j = 0; j < col; j++) {printf("| %c ",board[i][j]);}printf("|\n");if (i < row ) {for (int j = 0; j < col; j++) {printf("|___");}printf("|\n");}}printf("\n");
}
void PlayerMove(char board[ROW][COL], int row, int col) {while (1) {                            //若输入错误则一直循环int x, y;printf("请输入坐标:\n");scanf("%d %d", &x, &y);if (x <= row && x >= 1 && y <= col && y >= 1) {         //判断坐标是否合法if (board[x - 1][y - 1] == ' ')                     //玩家输入的坐标需要减1,因为数组从0开始,同时判断坐标是否被占用过{board[x - 1][y - 1] = '*';break;}else{printf("坐标被占用,请重新输入:\n");}}else{printf("坐标不合法,请重新输入:\n");}}
}
void ComputerMove(char board[ROW][COL], int row, int col) {while (1) {int x = rand() % row;  //生成合法的行坐标int y = rand() % col;  //生成合法的列坐标if (board[x][y]==' ') {board[x][y] = '#';break;}}
}char IsWin(char board[ROW][COL], int row, int col) {int x = 0;                   //计数器判断一行或一列是否遍历结束int sum1 = 0;                //判断*的数量int sum2 = 0;                //判断#的数量for (int i = 0; i < col; i++) {for (int j = 0; j < row; j++) {               //判断每行上是否有连续三个相等的符号if (board[i][j] == '*') {sum1 += 1;}if (board[i][j] == '#') {sum2 +=1;}x++;if(x==row){                                 //一行统计完时if (sum1 == row) {                        //一行全是*return '*';}else if (sum2 == row) {return '#';                         //一行全是#}else {                                  //一行中未分胜负则计数器清零统计下一行sum1 = 0;                         sum2 = 0;x = 0;}}}}for (int j = 0; j < row; j++) {for (int i = 0; i < col; i++) {                //行未分出胜负,则开始统计列if (board[i][j] == '*') {sum1 += 1;}if (board[i][j] == '#') {sum2 += 1;}x++;if (x == row) {                             //一列统计完时if (sum1 == row) {                        //一列全是*return '*';}else if (sum2 == row) {return '#';                         //一列连全是#}else {                                  //一列中未分胜负则计数器清零统计下一列sum1 = 0;sum2 = 0;x = 0;}}}}for (int i = 0; i < col; i++) {for (int j = 0; j < row; j++) {                 //行列都为分出胜负统计主对角线元素if (i == j) {if (board[i][j] == '*') {sum1 += 1;}else if (board[i][j] == '#') {sum2 += 1;}}}}if (sum1 == row) {return '*';                                     //主对角线上元素都为*}else if (sum2 == row) {return '#';                                     //主对角线上元素都为#}else {sum1 = 0;sum2 = 0;                                       //计数器清零统计副对角线}//查看副对角线上的元素,可以先复制一个横向翻转的临时棋盘board1,查看主对角线上的元素char board1[ROW][COL];for (int i = 0; i < row; i++) {for (int j = 0; j < col; j++) {board1[i][j] = board[i][col-j-1];}}//判断临时棋盘的主对角线元素for (int i = 0; i < col; i++) {for (int j = 0; j < row; j++) {                 if (i == j) {if (board1[i][j] == '*') {sum1 += 1;}else if (board1[i][j] == '#') {sum2 += 1;}}}}if (sum1 == row) {return '*';                                     //主对角线上元素都为*}else if (sum2 == row) {return '#';                                     //主对角线上元素都为#}else {sum1 = 0;sum2 = 0;                                       //计数器清零}//查看是否为平局,则是以上都判断完后,棋盘是否满了,满了则返回‘D’,不满则继续for (int i = 0; i < row; i++) {for (int j = 0; j < col; j++) {if (board[i][j] == ' ') {return 'C';                     //有地方是空的,游戏还未出结果}}}return 'D';                                 //棋盘满了且没分出胜负,则为平局
}

运行结果(以3子为例):

以上为C语言简易实现n子棋的内容

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

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

相关文章

XGBoost(eXtreme Gradient Boosting)

什么是机器学习 XGBoost&#xff08;eXtreme Gradient Boosting&#xff09;是一种梯度提升树算法&#xff0c;它在梯度提升框架的基础上引入了一些创新性的特性&#xff0c;以提高模型性能和训练速度。XGBoost在解决结构化数据的分类和回归问题上表现出色&#xff0c;成为许多…

【OpenCV学习笔记05】- 鼠标作为画笔

这是对于 OpenCV 官方文档的 GUI 功能的学习笔记。学习笔记中会记录官方给出的例子&#xff0c;也会给出自己根据官方的例子完成的更改代码&#xff0c;同样彩蛋的实现也会结合多个知识点一起实现一些小功能&#xff0c;来帮助我们对学会的知识点进行结合应用。 如果有喜欢我笔…

leetcode-相同的树

100. 相同的树 使用递归的方法 # Definition for a binary tree node. # class TreeNode: # def __init__(self, val0, leftNone, rightNone): # self.val val # self.left left # self.right right class Solution:def isSameTree(self, p: …

QTAV的编译、使用及遇到的问题

最近有个功能需求&#xff0c;界面可以直播显示某个特定摄像头的实时画面&#xff0c;通过对Qt本身的 QMiediaPlayer,和其他一些在网上找到的组件做了一些对比&#xff0c;最终选择了QtAv作为我们的组件使用。 QtAV 是一个基于 Qt 和 FFmpeg 的跨平台、高性能多媒体播放框架。…

CSAPP - string_length反汇编

虽然先前已经把 phase_1 和 phase_2 做出来了&#xff0c; 但其实是参考了网络上的答案&#xff0c; 仅仅是大概知道了关键汇编代码。但其实并没有真的懂。为啥呢&#xff1f;因为很多模棱两可的地方是靠猜测的&#xff0c;而猜测是脆弱的。 重新看 phase_1&#xff0c; 第一个…

用win系统搭建Minecraft世界服务器,MC开服教程,小白开服教程

雨云VPS用Windows系统搭建我的世界世界服务器&#xff0c;Minecraft开服教程&#xff0c;小白开服教程&#xff0c;MC 1.19.4版本服务器搭建教程。 此教程使用 Mohist 1.19.4 服务端&#xff0c;此服务端支持Forge模组和Bukkit/Spigot/Paper插件&#xff0c;如果需要开其他服务…

Linux 开启Swap交换内存

Linux 开启Swap交换内存 Linux 开启Swap交换内存 Linux 开启Swap交换内存 由于阿里云服务器内存空间有点小&#xff0c;但是又不能加内存&#xff0c;则想到使用Swap交换空间&#xff0c;用硬盘空间充当内存使用。 开启步骤&#xff0c;以4G为例&#xff1a; 1.创建交换文件…

K8S---kubectl top

一、简介 该命令类似于linux–top命令&#xff0c;用于显示node和pod的CPU和内存使用情况 二、命令行 /opt/kubernetes/bin/kubectl --kubeconfig /opt/kubernetes/conf/default-admin.kubeconfig top pod --help /opt/kubernetes/bin/kubectl --kubeconfig /opt/kube…

MySQL-索引回顾

索引是面试高频问答题&#xff0c;参考百度/CSDN/尚硅谷/黑马程序员/阿里云开发者社区&#xff0c;决定将索引知识回顾一下&#xff0c;忘记时&#xff0c;点开即可&#xff0c;时刻保持更新&#xff0c;事不宜迟&#xff0c;即刻享用。 索引概述 索引&#xff08;index&#…

transbigdata笔记:数据预处理

0 数据 使用 transbigdata/docs/source/gallery/data/TaxiData-Sample.csv at main ni1o1/transbigdata (github.com) 和transbigdata/docs/source/gallery/data/sz.json at main ni1o1/transbigdata (github.com) 0.1 导入库 import transbigdata as tbd import pandas …

VTK开发调试环境下载(VTK开发环境一步到位直接开发,无需自己配置编译 VS2017+Qt5.12.10+VTK)

一、无与伦比的优势 直接下载代码就可以调试的VTK代码仓库。 二、资源制作原理 这个资源根据VTK源码 编译出动态库文件 pdb lib dll 文件&#xff08; x64 debug &#xff09; 并将这两者同时放在一个代码仓库里&#xff0c;下载就能用。 三、使用方法&#xff08;vtk-so…

【Java代码审计】硬编码密码篇

【Java代码审计】硬编码密码篇 1.硬编码2.案例3.修复方案 1.硬编码 硬编码密码是指在系统中采用明文的形式存储密码&#xff0c;通常会导致严重的身份验证失败&#xff0c;这对于系统管理员而言可能很难检测到&#xff0c;一旦检测到&#xff0c;也很难修复。硬编码密码会造成…

使用rembg库提取图像前景(移除图像背景),并构建web应用

1、图像中的前景与背景 在深度学习图像处理领域中&#xff0c;图像内容可以被定义为前景与背景两部分&#xff0c;其中感兴趣图形的被定义为前景&#xff0c;不感兴趣区域的背景。如在目标检测中&#xff0c;被框出来的目标则被定义为前景。此外&#xff0c;前景识别也可以理解…

网络安全B模块(笔记详解)- nmap扫描渗透测试

nmap扫描渗透测试 1.通过BT5对服务器场景Linux进行TCP同步扫描 (使用工具Nmap,使用参数n,使用必须要使用的参数),并将该操作使用命令中必须要使用的参数作为Flag提交; Flag:sS 2.通过BT5对服务器场景Linux进行TCP同步扫描 (使用工具Nmap,使用参数n,使用必须要使用的参数…

oracle19c容器数据库data dump 数据泵传输数据(1)--pdb导pdb

目录 1.在pdb1创建实验环境 2.创建目标数据库pdb2 3.开始从pdb1全库导出 4.开始导入到pdb2 5. 解决报错&#xff1a;添加在pdb2添加users表空间 我們要記住一点&#xff1a;如果是全库导出导入的话&#xff0c;目标数据库没有的表空间我们要事先创建&#xff1a;不然就会导…

网工内推 | 高级网工,H3C认证优先,朝九晚六,周末双休

01 万德 招聘岗位&#xff1a;高级网络工程师 职责描述&#xff1a; 1、项目交付&#xff1a;项目管理和交付&#xff0c;包括项目前期的规划、实施以及后期的运维支持、项目验收等。 2、技术支持&#xff1a;为客户及合作伙伴提供网上问题远程和现场支持&#xff1b;对公司内…

移动通信系统关键技术多址接入MIMO学习(8)

1.Multiple-antenna Techniques多天线技术MIMO&#xff0c;从SISO到SIMO到MISO到如今的MIMO&#xff1b; 2.SIMO单发多收&#xff0c;分为选择合并、增益合并&#xff1b;SIMO&#xff0c;基站通过两路路径将信号发送到终端&#xff0c;因为终端接收到的两路信号都是来自同一天…

旋转的表示

欢迎访问我的博客首页。 旋转的表示 1.旋转轴的性质2.罗德里格斯公式3.右雅可比矩阵 三维空间内的旋转可以由三维旋转向量 n θ \bm n \theta nθ 表示。其中&#xff0c;单位向量 n \bm n n 表示旋转轴&#xff0c; θ \theta θ 表示旋转角度。旋转向量由一个轴和一个角表示…

大模型训练营Day3 基于 InternLM 和 LangChain 搭建你的知识库

本次的授课人是一个提示词开发项目的负责人。下面一起进入本期课程吧》 本次课程内容主要如下&#xff1a; 开篇交代了大模型的局限性&#xff0c;然后引出主题&#xff1a; 简单总结&#xff0c;大模型是根据数据集训练&#xff0c;很难使用具有实时性的数据进行重新训练&am…

Hana 实时数据同步优化(3)

简述 CloudCanal 近期对 Hana 源端链路做了新一轮优化&#xff0c;优化点主要来自用户实际场景使用&#xff0c;这篇文章简要做下分享。 本轮优化主要包含: 新增任务级增量表新增增量表定时清理能力新增增量表表结构自动演进能力任务延迟判定优化Hana 1.x 的兼容产品化和文档…