人生第一款人机对战小程序——三子棋(五千字无敌详解还有图)

2023-10-30 09:59

本文主要是介绍人生第一款人机对战小程序——三子棋(五千字无敌详解还有图),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

🍉说在前面

🍉程序整体思路

🍉定义一个头文件

🍉来一个漂亮的界面

🍉main函数的大结构

🍉Game()函数的大结构

🍇给二维数组初始化

🍇打印一个棋盘

🍇玩家下棋部分

🍇判断输赢的函数

🍇《关于我将这些函数整合在一起这些事》

🍉写在最后


🍉说在前面

五一小长假之后还是要继续努力了呢,来个鸡汤🍜

🍉程序整体思路

我们要写的是一个三子棋小游戏,也就是我们常说的井字棋。接下来盘一盘,我们需要那些部分来写成这个小程序。

  1. 下棋需要棋盘——打印一个棋盘
  2. 下棋需要两个人——分别实现玩家和电脑的下棋操作
  3. 需要记录我们下的棋子——利用一个二维字符数组记录
  4. 需要有人判断谁赢了——写一个函数判断赢还是输还是平局
  5. 写的程序需要好看——编写一个菜单界面

这就是我们在写这个小游戏前想到需要写的东西,分开成每个部分其实都不难(循环、二位数组、判断语句)就是小小的组合了一下。

这里我们采用的是分头文件、源文件的写法,顺便附上教程链接:程序的分文件写法及注意事项

 在这里我构建了三个文件,game.h文件存放函数的声明、game.c函数存放函数的定义、三子棋.c文件是程序的实现必不可少的部分。(这里不懂也不影响往下面看的)

🍉定义一个头文件

#include<stdio.h>
#include<time.h>//下文解释,实现人机将会用到
#include<stdlib.h>//实现人机部分用到
#define ROW 3//定义棋盘的行
#define COL 3//定义棋盘的列
void menu();//给菜单的定义
void Game();//给游戏函数的定义
void InitBoard(char Broad[ROW][COL], int row, int col);//定义初始化储存二维数组的函数
void BoardDis(char broad[ROW][COL],int row,int col);//定义打印棋盘
void Player(char broad[ROW][COL], int row, int col);//定义玩家下棋的函数
void Com(char broad[ROW][COL],int row, int col);//电脑下棋的函数
char Is_win(char broad[ROW][COL], int row, int col);//判断那方获胜的函数

 在这个文件里我们定义了我们写这个程序将会用到的定义,这样集中起来的一个好处是,这样程序的各个函数将会更直观的呈现出来。这里面的函数被定义出来后,将会在Game.c中声明

这个定义不是我一口气就想出来的,是一边写一边补充的。

 

🍉来一个漂亮的界面

void menu()//游戏菜单
{printf("*******************\n");printf("***** 1. play *****\n");printf("***** 0. exit *****\n");printf("*******************\n");
}

这一部分的作用是打印出一个游戏菜单。 这个函数写在game.c文件中。

 

🍉main函数的大结构

int main()
{srand((usigned int)time(NULL));//这个在电脑下棋那一段会用到,这里设置了一个随机数的种子int op = 0;do{menu();//菜单函数printf("请选择\n");scanf("%d", &op);switch (op){case 1:Game()//这个函数来将游戏实现break;case 0:printf("正在退出游戏\n");break;default:printf("输入错误,请重新输入\n");break;}} while (op);//当你选择了0,循环自动结束return 0;
}

我们的main函数值负责是否进行游戏的功能。

这里用到了switch选择结构来实现menu()函数的选择,并加了一个循环来实现重复游戏的效果。玩家在进行了一局游戏之后可以继续选择是否继续玩这个游戏。

🍉Game()函数的大结构

void Game()
{char Board[ROW][COL]={0};//储存旗子的二维数组//给二维数组初始化的函数//打印棋盘while(1){//玩家下棋的函数//判断输赢的函数//打印棋盘的函数//电脑下棋的函数//判断输赢的函数//打印棋盘的函数}
}

在这个函数中,我们会实现三子棋游戏的游戏部分。玩家下棋、电脑下棋、棋盘打印、判断输赢都会整合在这里。

🍇给二维数组初始化

void InitBoard(char board[ROW][COL], int row, int col)
{int i = 0; int j = 0;for (i; i < row; i++){for (j=0; j < col; j++){board[i][j] = ' ';}}return;}

在开始下棋前,我们将储存棋子的二维数组初始化为‘ ’(空格),这样就可以实现下面这样的效果。

 

 每次打印出来的在我们的棋盘上其实打印的是一个空格。

🍇打印一个棋盘

我们来观察这个棋盘,实质上它是两个空格夹着我们的棋子的如下图的位置

 

void BroadDis(char board[ROW][COL], int row, int col)//我们需要知道要摆放什么棋子,以及棋盘的大小
{int i = 0, j = 0;for (i; i < row; i++){for (j=0; j < col; j++){printf(" %c ", board[i][j]);if (j < COL-1)printf("|");}printf("\n");if(i<ROW-1)for (j=0;j<col;j++){printf("---");if (j < col - 1)printf("|");}printf("\n");}
}

 

if (j < COL-1)printf("|");

这一部分是一个巧妙的设计,为了防止多打一个‘|’,我们在最后一次循环阻止了打印‘|’。下面这一部分也是同理:

if(i<ROW-1)for (j=0;j<col;j++){printf("---");if (j < col - 1)printf("|");}

有些同学很好奇为什么我们要在头文件宏定义一个ROW、COL,这样我们想改变棋盘大小的时候只需要改变头文件里面的数据就行了,比如:我把ROW、COL改成了10,棋盘就会变成下面这样:

这么大写个五子棋都不过分吧😁。

🍇玩家下棋部分

开始下棋咯!

void Player(char broad[ROW][COL], int row, int col)//我们要用到二维数组来储存棋子
{int x = 0, y = 0;printf("玩家下棋\n");while (1)//在玩家棋子下位置前循环不会结束{scanf("%d %d", &x, &y);if (x <= row && y <= col && x > 0 && y > 0)//我们的棋子不能越界{if (broad[x - 1][y - 1] != ' ')//我们的棋子不能占用已经棋子的位置printf("该位置被占用\n");else{broad[x-1][y-1] = '#';break;//玩家位置下对了,训话结束}}elseprintf("输入的位置不合法\n");}
}

几乎每一行都进行了标注,贼良心🎉。

🍇电脑下棋了

电脑下棋,我们这里采用随机数的方式来模拟电脑下棋。随机数的操作方法可以看看这篇文章的前小半部分(极简随机数)。

void Com(char broad[ROW][COL], int row, int col)
{printf("电脑下棋\n");int x = 0, y = 0;while (1){x = rand() % ROW;//给下的行数赋一个值(取余是为了控制x的大小在0~2,下同)y = rand() % COL;//给下的列数赋一个值if (broad[x][y] == ' ' ){broad[x][y] = '*';//电脑下的棋子是'*'break;//电脑下对了才退出循环}}return;
}

 每下一步,我们就打印一次棋盘,现在可以来看看效果了:

 

🍇判断输赢的函数

我这里的思路是写一个函数判断输赢。

  • 玩家获胜——函数返回’#‘
  • 电脑获胜——函数返回’*‘
  • 平局——函数返回’P‘
  • 棋盘还没下满——函数返回’N‘

那我们这次定义的函数就需要是char类型的了。

char Is_win(char broad[ROW][COL], int row, int col)//把我们的棋盘导进去判断输赢
{//在行上判断有没有人赢了int i = 0, j = 0;for (i = 0; i < ROW; i++){//只要第一个等于第二个、第二个等于第三个就可以判定为赢,且这三个不能是空格,下同if (broad[i][0] == broad[i][1] && broad[i][1] == broad[i][2] && broad[i][0] != ' ')return broad[i][0];//谁赢了就返回谁的棋子}//在列上判断输赢for (j = 0; j < COL; j++){if (broad[0][j] == broad[1][j] && broad[1][j] == broad[2][j] && broad[0][j] != ' '){return broad[0][j];}}//对角线情况比较少,我就直接列出来了if (broad[0][0] == broad[1][1] && broad[1][1] == broad[2][2]&&broad[1][1]!=' ' || broad[0][2] == broad[1][1] && broad[1][1] == broad[2][0] && broad[1][1] != ' '){return broad[1][1];}int leap = 1;//假设棋盘满了//是否满了,如果我们的棋盘没满就还需要继续比赛for (i = 0; i < row; i++){for (j = 0; j < col; j++){if (broad[i][j] == ' '){leap = 0;break;}}}if (leap == 1)//棋盘满了就返回Preturn 'P';return 'N';
}

 这里也有注释,应该能看懂的,要是还没看懂的话,随时私聊我🌹。

🍇《关于我将这些函数整合在一起这些事》

void Game()
{char key = {0};//存放函数返回值的变量char Broad[ROW][COL] = { 0 };//定义一个存放棋子的二维数组InitBroad(Broad, ROW, COL);//数组初始化BroadDis(Broad,ROW,COL);//打印棋盘while (1)//一直下棋,知道分出胜负或平局{Player(Broad, ROW, COL);//玩家下棋BroadDis(Broad, ROW, COL);//打印棋盘key=Is_win(Broad, ROW, COL);//判断下棋结果if (key != 'N')//棋盘没满的话我们继续下棋break;//函数要是返回了N以外的字符我们停止循环Com(Broad,ROW,COL);BroadDis(Broad, ROW, COL);key=Is_win(Broad, ROW, COL);if (key != 'N')break;}//判断一下谁赢了if (key == '#')printf("玩家获胜\n");else if (key == '*')printf("电脑获胜\n");elseprintf("平局\n");return;
}

再次良心的注释了一切。

🍉写在最后

写博客写道一半,突发奇想,拉着室友去外面骑小电驴逛街去了,溜达完回来写完博客已经凌晨一点了哈哈😂,家人们晚安。

尼尔叔叔:你自己试试做吧。

 

 

加油!!评论评论鼓励鼓励博主吧 

 

(小福利) 

这篇关于人生第一款人机对战小程序——三子棋(五千字无敌详解还有图)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/307140

相关文章

Redis中6种缓存更新策略详解

《Redis中6种缓存更新策略详解》Redis作为一款高性能的内存数据库,已经成为缓存层的首选解决方案,然而,使用缓存时最大的挑战在于保证缓存数据与底层数据源的一致性,本文将介绍Redis中6种缓存更... 目录引言策略一:Cache-Aside(旁路缓存)策略工作原理代码示例优缺点分析适用场景策略二:Re

使用Python创建一个功能完整的Windows风格计算器程序

《使用Python创建一个功能完整的Windows风格计算器程序》:本文主要介绍如何使用Python和Tkinter创建一个功能完整的Windows风格计算器程序,包括基本运算、高级科学计算(如三... 目录python实现Windows系统计算器程序(含高级功能)1. 使用Tkinter实现基础计算器2.

Java注解之超越Javadoc的元数据利器详解

《Java注解之超越Javadoc的元数据利器详解》本文将深入探讨Java注解的定义、类型、内置注解、自定义注解、保留策略、实际应用场景及最佳实践,无论是初学者还是资深开发者,都能通过本文了解如何利用... 目录什么是注解?注解的类型内置注编程解自定义注解注解的保留策略实际用例最佳实践总结在 Java 编程

MySQL数据库约束深入详解

《MySQL数据库约束深入详解》:本文主要介绍MySQL数据库约束,在MySQL数据库中,约束是用来限制进入表中的数据类型的一种技术,通过使用约束,可以确保数据的准确性、完整性和可靠性,需要的朋友... 目录一、数据库约束的概念二、约束类型三、NOT NULL 非空约束四、DEFAULT 默认值约束五、UN

Python使用Matplotlib绘制3D曲面图详解

《Python使用Matplotlib绘制3D曲面图详解》:本文主要介绍Python使用Matplotlib绘制3D曲面图,在Python中,使用Matplotlib库绘制3D曲面图可以通过mpl... 目录准备工作绘制简单的 3D 曲面图绘制 3D 曲面图添加线框和透明度控制图形视角Matplotlib

MySQL中的分组和多表连接详解

《MySQL中的分组和多表连接详解》:本文主要介绍MySQL中的分组和多表连接的相关操作,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友一起看看吧... 目录mysql中的分组和多表连接一、MySQL的分组(group javascriptby )二、多表连接(表连接会产生大量的数据垃圾)MySQL中的

Java 实用工具类Spring 的 AnnotationUtils详解

《Java实用工具类Spring的AnnotationUtils详解》Spring框架提供了一个强大的注解工具类org.springframework.core.annotation.Annot... 目录前言一、AnnotationUtils 的常用方法二、常见应用场景三、与 JDK 原生注解 API 的

redis中使用lua脚本的原理与基本使用详解

《redis中使用lua脚本的原理与基本使用详解》在Redis中使用Lua脚本可以实现原子性操作、减少网络开销以及提高执行效率,下面小编就来和大家详细介绍一下在redis中使用lua脚本的原理... 目录Redis 执行 Lua 脚本的原理基本使用方法使用EVAL命令执行 Lua 脚本使用EVALSHA命令

SpringBoot3.4配置校验新特性的用法详解

《SpringBoot3.4配置校验新特性的用法详解》SpringBoot3.4对配置校验支持进行了全面升级,这篇文章为大家详细介绍了一下它们的具体使用,文中的示例代码讲解详细,感兴趣的小伙伴可以参考... 目录基本用法示例定义配置类配置 application.yml注入使用嵌套对象与集合元素深度校验开发

Python中的Walrus运算符分析示例详解

《Python中的Walrus运算符分析示例详解》Python中的Walrus运算符(:=)是Python3.8引入的一个新特性,允许在表达式中同时赋值和返回值,它的核心作用是减少重复计算,提升代码简... 目录1. 在循环中避免重复计算2. 在条件判断中同时赋值变量3. 在列表推导式或字典推导式中简化逻辑