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

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

相关文章

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

OpenHarmony鸿蒙开发( Beta5.0)无感配网详解

1、简介 无感配网是指在设备联网过程中无需输入热点相关账号信息,即可快速实现设备配网,是一种兼顾高效性、可靠性和安全性的配网方式。 2、配网原理 2.1 通信原理 手机和智能设备之间的信息传递,利用特有的NAN协议实现。利用手机和智能设备之间的WiFi 感知订阅、发布能力,实现了数字管家应用和设备之间的发现。在完成设备间的认证和响应后,即可发送相关配网数据。同时还支持与常规Sof

JAVA智听未来一站式有声阅读平台听书系统小程序源码

智听未来,一站式有声阅读平台听书系统 🌟&nbsp;开篇:遇见未来,从“智听”开始 在这个快节奏的时代,你是否渴望在忙碌的间隙,找到一片属于自己的宁静角落?是否梦想着能随时随地,沉浸在知识的海洋,或是故事的奇幻世界里?今天,就让我带你一起探索“智听未来”——这一站式有声阅读平台听书系统,它正悄悄改变着我们的阅读方式,让未来触手可及! 📚&nbsp;第一站:海量资源,应有尽有 走进“智听

6.1.数据结构-c/c++堆详解下篇(堆排序,TopK问题)

上篇:6.1.数据结构-c/c++模拟实现堆上篇(向下,上调整算法,建堆,增删数据)-CSDN博客 本章重点 1.使用堆来完成堆排序 2.使用堆解决TopK问题 目录 一.堆排序 1.1 思路 1.2 代码 1.3 简单测试 二.TopK问题 2.1 思路(求最小): 2.2 C语言代码(手写堆) 2.3 C++代码(使用优先级队列 priority_queue)

K8S(Kubernetes)开源的容器编排平台安装步骤详解

K8S(Kubernetes)是一个开源的容器编排平台,用于自动化部署、扩展和管理容器化应用程序。以下是K8S容器编排平台的安装步骤、使用方式及特点的概述: 安装步骤: 安装Docker:K8S需要基于Docker来运行容器化应用程序。首先要在所有节点上安装Docker引擎。 安装Kubernetes Master:在集群中选择一台主机作为Master节点,安装K8S的控制平面组件,如AP

EMLOG程序单页友链和标签增加美化

单页友联效果图: 标签页面效果图: 源码介绍 EMLOG单页友情链接和TAG标签,友链单页文件代码main{width: 58%;是设置宽度 自己把设置成与您的网站宽度一样,如果自适应就填写100%,TAG文件不用修改 安装方法:把Links.php和tag.php上传到网站根目录即可,访问 域名/Links.php、域名/tag.php 所有模板适用,代码就不粘贴出来,已经打

跨系统环境下LabVIEW程序稳定运行

在LabVIEW开发中,不同电脑的配置和操作系统(如Win11与Win7)可能对程序的稳定运行产生影响。为了确保程序在不同平台上都能正常且稳定运行,需要从兼容性、驱动、以及性能优化等多个方面入手。本文将详细介绍如何在不同系统环境下,使LabVIEW开发的程序保持稳定运行的有效策略。 LabVIEW版本兼容性 LabVIEW各版本对不同操作系统的支持存在差异。因此,在开发程序时,尽量使用

嵌入式Openharmony系统构建与启动详解

大家好,今天主要给大家分享一下,如何构建Openharmony子系统以及系统的启动过程分解。 第一:OpenHarmony系统构建      首先熟悉一下,构建系统是一种自动化处理工具的集合,通过将源代码文件进行一系列处理,最终生成和用户可以使用的目标文件。这里的目标文件包括静态链接库文件、动态链接库文件、可执行文件、脚本文件、配置文件等。      我们在编写hellowor

LabVIEW FIFO详解

在LabVIEW的FPGA开发中,FIFO(先入先出队列)是常用的数据传输机制。通过配置FIFO的属性,工程师可以在FPGA和主机之间,或不同FPGA VIs之间进行高效的数据传输。根据具体需求,FIFO有多种类型与实现方式,包括目标范围内FIFO(Target-Scoped)、DMA FIFO以及点对点流(Peer-to-Peer)。 FIFO类型 **目标范围FIFO(Target-Sc

019、JOptionPane类的常用静态方法详解

目录 JOptionPane类的常用静态方法详解 1. showInputDialog()方法 1.1基本用法 1.2带有默认值的输入框 1.3带有选项的输入对话框 1.4自定义图标的输入对话框 2. showConfirmDialog()方法 2.1基本用法 2.2自定义按钮和图标 2.3带有自定义组件的确认对话框 3. showMessageDialog()方法 3.1