【C语言】小游戏-三字棋

大家好,我是深鱼~


目录

一、游戏介绍

 二、文件分装

 三、代码实现步骤

1.制作简易游戏菜单

 2.初始化棋盘

3.打印棋盘

4.玩家下棋

5.电脑随机下棋

6.判断输赢

7.判断棋盘是否满了

 四、完整代码

game.h(相关函数的声明,整个代码要引用的头文件以及宏定义)

game.c(实现游戏的相关函数)

test.c(:整个游戏相关的测试)


一、游戏介绍

《三子棋》是一款古老的民间传统游戏,又被称为黑白棋、圈圈叉叉棋、井字棋、一条龙、九宫棋等。游戏分为双方对战,双方依次在9宫格棋盘上摆放棋子,率先将自己的三个棋子连成一条线的一方则视为胜利者


 二、文件分装

源文件:内含函数实现,变量定义等内容

头文件:内含函数声明、宏定义、结构体定义等内容 

实现这个三子棋,我创建了三个文件

源文件:

test.c:整个游戏相关的测试

game.c:实现游戏的相关函数

头文件:

game.h: 相关函数的声明,整个代码要引用的头文件以及宏定义


 三、代码实现步骤

1.制作简易游戏菜单

(1)定义一个menu函数来打印游戏菜单,菜单中需要有游戏菜单选项:1.开始游戏  0.退出游戏,当我们输入1就进入游戏,0就退出游戏

(2)考虑到玩家可能玩一局以后还想继续玩,就用do while循环打印菜单,while(input),当输入的值为0,就跳出循环(退出游戏),如果输入非0值就再次进入循环(再次选择)

(3)玩家选择了进入游戏还是退出游戏需要判断,就用switch case语句来进行判断

test.c

#include"game.h"
void menu()
{printf("*****************************************\n");printf("**********      1.开始游戏      *********\n");printf("**********      0.退出游戏      *********\n");printf("*****************************************\n");}int main()
{int input = 0;srand((unsigned int)time(NULL));do//实现玩多盘游戏{ menu();//打印菜单printf("请选择1/0:>\n");scanf("%d", &input);//选择进入游戏/退出游戏switch (input)//判断输入input的值决定是进入游戏还是退出游戏{case 1:game();//三子棋游戏的实现break;case 0:printf("退出游戏\n");break;default:printf("选择错误,请重新选择\n");break;}} while (input);//当input为0,跳出循环;为1再玩一盘游戏(重新进入循环);为其他值进入循环重新选择return 0;
}

考虑到game.c和test.c可能都要用printf函数的头文件#include<stdio.h>,我们直接把这个头文件写进game.h,那么test.c这个文件只需引用#inlclude"game.h"即可

系统头文件一般用<>

用户自己定义的则可以使用" ",加快搜索速度

效果展示:

选择1,进入game()

选择0,退出游戏:

选择非1非0(选择错误),重新选择 

 


 2.初始化棋盘

(1)首先我们需要定义一个二维数组来存放数据,可以采用宏定义

宏定义:方便程序的修改,如果我们要改变棋盘的大小,直接修改宏定义的数字即可

(2)初始化棋盘我们定义一个InitBoard(board)函数,将棋盘全部初始化为' '

game.h

//宏定义:
//为了方便直接改变棋盘的大小,在头文件里面定义(ROW和COL就是常量)
#define ROW 3 //没有逗号     //行
#define COL 3                //列//初始化数组为空格
void InitBoard(char board[ROW][COL]);

game.c

//初始化数组为空格
void InitBoard(char board[ROW][COL])
{for (int i = 0; i <ROW; i++){for (int j = 0; j <COL; j++){board[i][j] = ' ';//全部初始化为空格}}
}

3.打印棋盘

我们想打印这样打一个棋盘:用分隔线分开方便观察

(1)先打印数据(最后一个不用打印‘|’ )

(2)再打印分割行(最后一行不用打印)

game.h

//打印棋盘
void DisplayBoard(char board[ROW][COL]);

 game.c

//打印棋盘
void DisplayBoard(char board[ROW][COL])
{for (int i = 0; i < ROW; i++){//先打印数据for (int j = 0; j < COL; j++){printf(" %c ", board[i][j]);if(j<COL-1)//一行的最后一个不打印printf("|");}printf("\n");//再打印分割行(最后一行不打印)if (i < ROW - 1)//最后一行不打印{for (int j = 0; j < COL; j++){printf("---");if (j < COL - 1)//也是一行的最后一个不打印printf("|");}printf("\n");}}
}

4.玩家下棋

注意:

(1)玩家下棋输入坐标要在1-3(如果不在这个范围要重新输入),而不是数组下标的范围0-2,所以放入坐标是输入坐标分别-1得到的结果放入

(2)玩家下棋我们需要考虑坐标是否已经被占用

(3)字符的比较可以用==,既可用于常量也可用于变量比较

       字符串的比较要用strcmp函数,并且只能用于变量比较

game.h

//打印棋盘
void DisplayBoard(char board[ROW][COL]);

 game.c

//玩家下棋
void PlayerMove(char board[ROW][COL])
{int x = 0;int y = 0;printf("玩家下棋\n");while (1){printf("请输入要下棋的坐标:>\n");scanf("%d %d", &x, &y);if (x >= 1 && x <= ROW && y >= 1 && y <= COL)//下棋的坐标必须在二维数组内部{if (board[x - 1][y - 1] == ' ')//1.输入的横纵坐标-1才是数组的下标  2.字符串的比较不能用==,但是这只是一个字符空格而已{board[x - 1][y - 1] = '*';break;}else{printf("该坐标已被占用,请输入其他坐标\n");}}else{printf("坐标非法,请重新输入\n");}}
}

5.电脑随机下棋

电脑随机下棋就需要随机生成两个随机数,但是这两个数要在棋盘范围内,那就对生成的随机数进行取模x = rand() % ROW;产生的数为0-(ROW-1),这个时候就不用-1了,因为电脑下棋不用我们自己输入,它就以数组的形式下棋就行

这里介绍三个函数

rand():

头文件:#include<stdlib.h>

这个函数返回的是0-rand max之间的值(0-32767)

在使用rand函数之前,我们要求调用srand函数,这样才能实现rand的功能

srand():

头文件:#include<stdlib.h>

要想生成一个随机数,还得给srand一个随机数(要想rand函数产生的随机数不同,就需要给srand函数输入一个不断改变的值(如果只给一个常数,那么rand给出的随机数就是同一个值,不会改变))

srand的参数需要是unsigned int(无符号整形)eg:srand((unsigned int)time(NULL));

time():

头文件:#include<time.h>

原理:时间戳:把不停在改变的时间转化为数字返回,从而给srand一个随机值

game.h

//电脑下棋
void ComputerMove(char board[ROW][COL]);

 game.c

//电脑随机下棋
void ComputerMove(char board[ROW][COL]) 
{int x = 0;int y = 0;printf("电脑下棋\n");while (1){x = rand() % ROW;//产生的数为0-(ROW-1)y = rand() % COL;if (board[x][y] == ' ')//如果这个位置已经有棋子了,再进入循环,重新生成x和y直到下了一个棋子为止{board[x][y] = '#';break;}}
}

6.判断输赢

判断输赢:设置四种结果:
玩家赢-返回*
电脑赢-#
平局-Q
游戏继续—C

game.h

//判断输赢
char IsWin(char board[ROW][COL]);

 game.c

//判断输赢:四种结果:
//玩家赢-返回*
//电脑赢-#
//平局-Q
//游戏继续—C
char IsWin(char board[ROW][COL])
{for (int i = 0; i < ROW; i++){if (board[i][0] == board[i][1] && board[i][0] == board[i][2] && board[i][0] != ' ')//注意要排除三个空格的情况return board[i][0];//谁的棋子连着三个就返回谁的棋子}for (int i = 0; i < COL; i++){if (board[0][i] == board[1][i] && board[0][i] == board[2][i] && board[0][i] != ' ')//注意要排除三个空格的情况return board[0][i];//谁的棋子连着三个就返回谁的棋子}if (board[0][0] == board[1][1] && board[0][0] == board[2][2] && board[0][0] != ' ')return board[0][0];if (board[0][2] == board[1][1] && board[1][1] == board[2][0] && board[1][1] != ' ')return board[1][1];if (IsFull(board))return 'Q';//如果棋盘满了,就表示平局,返回Qreturn 'C';//否则就继续下棋}

7.判断棋盘是否满了

遍历二维数组,只要有一个空格就继续下棋(没满),如果都不是空格,那就是表示满了

game.h

//判断棋盘是否满了
int IsFull(char board[ROW][COL]);

 game.c

//判断棋盘是否是满的
static int IsFull(char board[ROW][COL])//staic表示只在这个game.c的文件中使用这个函数
{for (int i = 0; i < ROW; i++){for (int j = 0; j < COL; j++){if (board[i][j] == ' ')//只要有一个空格就继续进行return 0;}}return 1;//如果都不是空格,那就是表示全满
}

 四、完整代码

game.h(相关函数的声明,整个代码要引用的头文件以及宏定义)

//头文件:内含函数声明、宏定义、结构体定义等内容#include<stdio.h>
#include<time.h>
#include<stdlib.h>//宏定义:
//为了方便直接改变棋盘的大小,在头文件里面定义(ROW和COL就是常量)
#define ROW 3 //没有逗号     //行
#define COL 3                //列//函数声明:
//初始化数组为空格
void InitBoard(char board[ROW][COL]);//row和col为真实的行和列
//打印棋盘
void DisplayBoard(char board[ROW][COL]);
//玩家下棋
void PlayerMove(char board[ROW][COL]);
//电脑下棋
void ComputerMove(char board[ROW][COL]);
//判断输赢
char IsWin(char board[ROW][COL]);
//判断棋盘是否满了
int IsFull(char board[ROW][COL]);

game.c(实现游戏的相关函数)

//实现游戏相关的代码
#include"game.h"//这里也要包含自己定义的头文件,不然ROW和COL宏定义就不起作用了//初始化数组为空格
void InitBoard(char board[ROW][COL])
{for (int i = 0; i <ROW; i++){for (int j = 0; j <COL; j++){board[i][j] = ' ';//全部初始化为空格}}
}//打印棋盘
void DisplayBoard(char board[ROW][COL])
{for (int i = 0; i < ROW; i++){//先打印数据for (int j = 0; j < COL; j++){printf(" %c ", board[i][j]);if(j<COL-1)//一行的最后一个不打印printf("|");}printf("\n");//再打印分割行(最后一行不打印)if (i < ROW - 1)//最后一行不打印{for (int j = 0; j < COL; j++){printf("---");if (j < COL - 1)//也是一行的最后一个不打印printf("|");}printf("\n");}}
}//玩家下棋
void PlayerMove(char board[ROW][COL])
{int x = 0;int y = 0;printf("玩家下棋\n");while (1){printf("请输入要下棋的坐标:>\n");scanf("%d %d", &x, &y);if (x >= 1 && x <= ROW && y >= 1 && y <= COL)//下棋的坐标必须在二维数组内部{if (board[x - 1][y - 1] == ' ')//1.输入的横纵坐标-1才是数组的下标  2.字符串的比较不能用==,但是这只是一个字符空格而已{board[x - 1][y - 1] = '*';break;}else{printf("该坐标已被占用,请输入其他坐标\n");}}else{printf("坐标非法,请重新输入\n");}}
}//电脑随机下棋
void ComputerMove(char board[ROW][COL]) 
{int x = 0;int y = 0;printf("电脑下棋\n");while (1){x = rand() % ROW;//产生的数为0-(ROW-1)y = rand() % COL;if (board[x][y] == ' ')//如果这个位置已经有棋子了,再进入循环,重新生成x和y直到下了一个棋子为止{board[x][y] = '#';break;}}
}//判断棋盘是否是满的
static int IsFull(char board[ROW][COL])//staic表示只在这个game.c的文件中使用这个函数
{for (int i = 0; i < ROW; i++){for (int j = 0; j < COL; j++){if (board[i][j] == ' ')//只要有一个空格就继续进行return 0;}}return 1;//如果都不是空格,那就是表示全满
}//判断输赢:四种结果:
//玩家赢-返回*
//电脑赢-#
//平局-Q
//游戏继续—C
char IsWin(char board[ROW][COL])
{for (int i = 0; i < ROW; i++){if (board[i][0] == board[i][1] && board[i][0] == board[i][2] && board[i][0] != ' ')//注意要排除三个空格的情况return board[i][0];//谁的棋子连着三个就返回谁的棋子}for (int i = 0; i < COL; i++){if (board[0][i] == board[1][i] && board[0][i] == board[2][i] && board[0][i] != ' ')//注意要排除三个空格的情况return board[0][i];//谁的棋子连着三个就返回谁的棋子}if (board[0][0] == board[1][1] && board[0][0] == board[2][2] && board[0][0] != ' ')return board[0][0];if (board[0][2] == board[1][1] && board[1][1] == board[2][0] && board[1][1] != ' ')return board[1][1];if (IsFull(board))return 'Q';//如果棋盘满了,就表示平局,返回Qreturn 'C';//否则就继续下棋}

test.c(:整个游戏相关的测试)

game函数的逻辑:

(1)初始化数组,并打印出来看看什么样子

(2)开始下棋,玩家先下,下完进行打印,打印完进行判断输赢,如果不是游戏继续,就直接跳出循环,输出游戏结果(玩家赢/电脑赢/平局)

(3)电脑下棋下棋的逻辑一样 

//源文件,内含函数实现,变量定义等内容#include"game.h"//系统头文件一般用<>;用户自己定义的则可以使用""
//整个游戏相关的测试
void menu()
{printf("*****************************************\n");printf("**********      1.开始游戏      *********\n");printf("**********      0.退出游戏      *********\n");printf("*****************************************\n");}void game()
{char ret = 0;char board[ROW][COL];//开始的时候,数组的内容应该全是空格InitBoard(board);//用于初始化数组为空格DisplayBoard(board);//用于打印棋盘//下棋while (1){PlayerMove(board);//玩家下棋放*DisplayBoard(board);//判断输赢ret=IsWin(board);if (ret != 'C'){break;//不是游戏继续就跳出循环,输出结果}ComputerMove(board);//电脑随机下棋放#DisplayBoard(board);//判断输赢ret = IsWin(board);if (ret != 'C'){break;}}if (ret == '*')printf("玩家赢\n");else if (ret =='#')printf("电脑赢\n");elseprintf("平局\n");
}
int main()
{int input = 0;srand((unsigned int)time(NULL));do//实现玩多盘游戏{ menu();//打印菜单printf("请选择1/0:>\n");scanf("%d", &input);//选择进入游戏/退出游戏switch (input)//判断输入input的值决定是进入游戏还是退出游戏{case 1:game();//三子棋游戏的实现break;case 0:printf("退出游戏\n");break;default:printf("选择错误,请重新选择\n");break;}} while (input);//当input为0,跳出循环;为1再玩一盘游戏(重新进入循环);为其他值进入循环重新选择return 0;
}

 本次C语言小游戏三子棋的内容就到此啦,有什么问题欢迎评论区或者私信交流,觉得笔者写的还可以,或者自己有些许收获的,麻烦铁汁们动动小手,给俺来个一键三连,万分感谢 ! 

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

如若内容造成侵权/违法违规/事实不符,请联系瞎胡闹网进行投诉反馈,一经查实,立即删除!

相关文章

Apache Paimon 在同程旅行的实践进展

摘要&#xff1a;本文整理自同程旅行大数据计算组负责人吴祥平&#xff0c;在 Apache Paimon Meetup 的分享。本篇内容主要分为四个部分&#xff1a; 1. Apache Paimon 引入 2. Apache Paimon 应用建设 3. Apache Paimon 优化实践 4. 未来规划和期待 Tips&#xff1a;点击「阅读…

Linux运维

目录 第一章、Linux概述 一、Linux的概念 二、Linux的特点 三、Linux VS Windows ​四、Linux的发展优势与存在问题-------不足 五、Linux常用发行版 六、CentOS简介 七、VMWare虚拟机简介 第二章、Linux初示 一、虚拟控制台 二 、Linux启动 &#xff08;1&#xf…

Linux极客汇总常用运维大全

一、基础篇 1、Linux版本 内核版本 发行版本 RedHat Enterprise Linux-&#xff08;公司级别&#xff0c;付费&#xff09; Fedora(组建一个社区&#xff0c;免费) CentOS&#xff08;基于红帽源代码编译的&#xff0c;免费&#xff09; Debian&#xff08;华丽界面&#xf…

Linux云服务器的使用,以及运行Python程序、相关Linux指令

目录 1、使用Linux云服务器的软件 1.1、MobaXterm_Personal 1.2、WindTerm 1.3、FileZilla FTP 2、Linux系统运行Python程序 3、Linux系统查看包、虚拟环境、安装包等 以下几个深度学习服务器都不错&#xff1a;智星云、AutoDL、恒源云 1、使用Linux云服务器的软件 1.1…

linux运维21

linux运维篇21 一、简述redis集群的实现原理二、基于redis5的redis cluster部署 一、简述redis集群的实现原理 工作原理&#xff1a;虽然redis有主从结构&#xff0c;但是无法解决只能单机写入数据的问题&#xff0c;无法实现分布式数据保存。而redis集群会预先分配16384个槽位…

linux运维15

linux运维篇15 一、实现基于MYSQL验证的vsftpd虚拟用户访问二、配置samba共享&#xff0c;实现/www目录共享三、使用rsyncinotify实现/www目录实时同步四、LVS调度算法总结五、LVS的跨网络DR实现 一、实现基于MYSQL验证的vsftpd虚拟用户访问 FTP服务器搭建&#xff1a; 数据库…

linux运维19

linux运维篇19 一、haproxy https实现二、总结tomcat的核心组件以及根目录结构三、tomcat实现多虚拟主机四、nginx实现后端tomcat的负载均衡调度五、简述memcached的工作原理 一、haproxy https实现 路由拓扑 后台web服务器搭建 LAMP架构看这个&#xff1a;LAMP yum install…

收集的 Linux VPS 在线重装系统脚本

因为 VPS 上预装的操作系统我并不习惯&#xff0c;所以打算重装一个。有的 VPS 服务商提供了较多种类的系统选择&#xff0c;有的却没有。如果你发现你希望重装的系统服务商没有提供&#xff0c;可以考虑自己安装。 本文内容 重装脚本 - 来自萌咖CentOS -> Debian 重装脚本 …

【运维】——服务器装Linux系统

一、用软碟通刻录系统光盘 1打开软碟通软件&#xff0c;打开文件-------选择Linux系统 2选择启动----写入硬盘映像 3开始写入硬盘映像&#xff0c;静等进度条走完即可。 二、开始安装Linux系统 1给主机插上刚刚刻录好的系统光盘&#xff0c;选择启动项 2设置信息&#xff1a;…

Linux入门---环境搭建(腾讯云服务器)、XShell远程登陆Linux

Linux环境搭建和远程登陆Linux Linux环境搭建XShell远程登陆Linux Linux环境搭建 主要有三种方式&#xff1a; 1.直接安装在物理机上&#xff0c;但是由于Linux桌面使用起来非常不友好&#xff0c;不推荐。 2.使用虚拟机软件&#xff0c;将Linux搭建在虚拟机上&#xff0c;但是…

Linux系统运维1 运维 项目研发 网站 服务器 计算机基础 Linux操作系统

运维的基本概念 运维行业前景 企业运作模式 四大部门 项目研发流程 职责描述&#xff1a; 运维的作用&#xff1a; 网站的相关概念 网站运行流程&#xff1a; IP<–>域名 重要概念&#xff1a; 服务器图片&#xff1a; 服务器&#xff1a;为用户提供服务的机器&…

linux运维14

linux运维篇14 一、简述CGI与FASTCGI区别二、 编译安装基于fastcgi模式的多虚拟主机的wordpress和discuz的LAMP架构三、通过loganalyzer展示数据库中的日志&#xff08;cgi模式&#xff09; 一、简述CGI与FASTCGI区别 CGI模式&#xff1a;当用户访问网站的动态资源时&#xff…

【从零开始学习JAVA | 第四十五篇】动态代理

目录 前言&#xff1a; 动态代理&#xff1a; 动态代理实现步骤&#xff1a; 动态代理的应用场景&#xff1a; 总结&#xff1a; 前言&#xff1a; 动态代理作为一种强大的编程技术&#xff0c;不仅为我们提供了灵活性和可扩展性&#xff0c;还为软件开发和系统设计带来了…

布隆过滤器在缓存系统中的实践

一. 背景 在业务开发中&#xff0c;在并发量很高的情况下&#xff0c;通常会使用缓存对系统查询性能进行优化&#xff0c;在缓存命中率很高的情况下&#xff0c;缓存的使用能够大幅提升系统查询性能。但是在缓存命中率非常低场景下&#xff0c;如果采用传统缓存读取模式&#…

Redisson_布隆过滤器

应用场景 去重 诞生背景 Java应用一般通过JDK自身提供的HashSet去重&#xff0c;通过contains()方法判断当前元素是否存在于Set中。该方式要求在调用contains()前&#xff0c;已经将数据列表加载到内存中&#xff08;即该方法基于内存存储实现判断功能&#xff09;。 缺点: 1.满…

【布隆过滤器】

我是&#x1f31f;廖志伟&#x1f31f;&#xff0c;一名&#x1f315;Java开发工程师&#x1f315;、&#x1f4dd;Java领域优质创作者&#x1f4dd;、&#x1f389;CSDN博客专家&#x1f389;、&#x1f339;幕后大佬社区创始人&#x1f339;。拥有多年一线研发经验&#xff0…

xmind用例数据上传至禅道

xmind格式参考&#xff0c;只需要在一级子主题填写对应用例模块ID&#xff0c;其余格式随意即可生成用例并直接上传到禅道&#xff1a; 代码里需填写禅道对应登录账号及用例所属产品 import requests import json import re import hashlib import pprint import threading fr…

认识相机

认识相机 在Threejs中相机的表示是THREE.Camera&#xff0c;它是相机的抽象基类&#xff0c;其子类有两种相机&#xff0c;分别是正投影相机THREE.OrthographicCamera和透视投影相机THREE.PerspectiveCamera。类图如下所示&#xff1a; 透视投影相机&#xff08;PerspectiveCam…

【项目实践】海康威视工业相机SDK开发小白版入门教程(VS2015+OpenCV4.5.1)

本文目录 前言怎么查找资料&#xff1f;数据手册例程 项目开发VS版本与OpenCV版本选择VS配置OpenCVVS添加MVS安装目录下的头文件和库VS项目开发 编程问题记录相机数据如何转换为OpenCV的Mat类型&#xff1f;函数不能修改全局指针变量&#xff1f;OpenCV运行报错“有未经处理的异…

Azure Kinect sdk 入门,简单使用深度相机

首先要安装azure Kinect dk传感器和人体跟踪的软件 先安装传感器&#xff1a;Azure-Kinect-Sensor-SDK/usage.md at develop microsoft/Azure-Kinect-Sensor-SDK GitHub 在这个网址里下载&#xff0c; 点击红笔画出来的地方&#xff0c;下载安装&#xff0c;记住安装路径&a…