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,一经查实,立即删除!

相关文章

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

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

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

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

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…

使用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,使用必须要使用的参数…

网工内推 | 高级网工,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;因为终端接收到的两路信号都是来自同一天…

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

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

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

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

Oracle数据库19c OCP 1z0-082考场真题解析第24题

考试科目&#xff1a;1Z0-082 考试题量&#xff1a;90 通过分数&#xff1a;60% 考试时间&#xff1a;150min 本文为云贝教育郭一军guoyJoe原创&#xff0c;请尊重知识产权&#xff0c;转发请注明出处&#xff0c;不接受任何抄袭、演绎和未经注明出处的转载。【云贝教育】Orac…

X3En【禾川】

地址: P9-00 波特率:【005】 DI: P4.0 滤波【微秒us】P4.1 DI1【1】使能 P4.2 DI2【2】复位 P4.3 DI3【13】禁止脉冲 P4.4 DI4【5】 P4.5 DI5【25】 P4.6 DI6【14】 P4.7 DI7【15】 P4.8 DI8【0】 P4.9 DI9【0】P4.11 DI1【逻辑】0低电…

Java并发之互斥一:管程

1、简单聊聊什么是管程模型 &#xff08;共享资源&#xff09;&#xff1a;定义一个共享变量&#xff0c;可以理解锁&#xff0c;令牌这类的东西&#xff08;互斥访问共享资源&#xff09;&#xff1a;获取这个锁、令牌的时候是排好队的&#xff0c;只允许单线程访问&#xff…

WPF XAML(一)

一、XAML的含义 问&#xff1a;XAML的含义是什么&#xff1f;为什么WPF中会使用XAML&#xff1f;而不是别的&#xff1f; 答&#xff1a;在XAML是基于XML的格式&#xff0c;XML的优点在于设计目标是具有逻辑性易读而且简单内容也没有被压缩。 其中需要提一下XAML文件在 Visu…

1886_emacs_v29中的行号配置

Grey 全部学习内容汇总&#xff1a; GitHub - GreyZhang/editors_skills: Summary for some common editor skills I used. emacs 29中的行号显示配置 行号显示 行号显示是一个编辑器中很常见的功能&#xff0c;我觉得这个功能的需求度可能因人群或者个人习惯而不同。对于只…

如何给字符串字段添加索引

MySQL是支持前缀索引的&#xff0c;可以定义字符串的一部分作为索引&#xff0c;如果创建索引的语句不指定前缀长度&#xff0c;那么索引就会包含整个字符串。 alter table SUser add index index1(email);alter table SUser add index index2(email(6)); 如上两个创建索引的语…

完整的模型训练套路(一、二、三)

搭建神经网络 model import torch from torch import nn#搭建神经网络 class Tudui(nn.Module):def __init__(self):super(Tudui, self).__init__()self.model nn.Sequential(nn.Conv2d(3, 32, 5, 1, 2),nn.MaxPool2d(2),nn.Conv2d(32, 32, 5, 1, 2),nn.MaxPool2d(2),nn.Conv…

Godot游戏引擎有啥优势

游戏开发是一个充满创造力和技术挑战的领域。选择适合的游戏引擎是开发过程中至关重要的决策。在众多的游戏引擎中&#xff0c;Godot以其许多的优势吸引着越来越多的开发者。下面简单的介绍一下Godot游戏引擎的优势。 免费和开源 Godot游戏引擎的免费和开源特性是优势之一。开…

动态规划day03

343. 整数拆分(第二次做还是没弄明白) 力扣题目链接(opens new window) 给定一个正整数 n&#xff0c;将其拆分为至少两个正整数的和&#xff0c;并使这些整数的乘积最大化。 返回你可以获得的最大乘积。 示例 1: 输入: 2输出: 1解释: 2 1 1, 1 1 1。 示例 2: 输入: …