C语言小游戏:三子棋

目录

🌍前言

🚅目录设计

💎游戏逻辑设置

⚔三子棋棋盘设计

⚔三子棋运行逻辑

👀怎么设置人下棋

👀怎么设置电脑下棋

✈如何判断输赢

✍结语


🌍前言

Hello,csdn的各位小伙伴你们好啊!这次小赵给大家带来的C语言小游戏是三子棋,三子棋也就是大家常玩的井字棋,这样的一个编程想必会让很多人都兴奋不已,好了好了那我们赶快开始吧。

🚅目录设计

目录的设计我们已经讲过好几遍了,还是最简单的那个,简单一个打印*加一些选项。然后用我们的switch函数来实现我们的选择,如输入1进入游戏,0跳出游戏,其他报错然后重新输入。

void meun()
{
	printf("*************************************************\n");
	printf("*********************0.exit**********************\n");
	printf("*********************1.play**********************\n");
	printf("*************************************************\n");
	printf("请输入你的选择:");
}
int main()
{
	int a = 0;
	do
	{
		meun();
		scanf("%d", &a);
		switch (a)
		{
		case 1:
			game();
			break;
		case 0:
			printf("游戏结束\n");
			break;
		default:
			printf("输入错误,请重新输入\n");
			break;
		}
	} while (a);

 在这里我们就不多说了,详细的讲解大家可以看看前面的扫雷和猜数字游戏。(大家可以关注小赵的专栏c语言小游戏哦!

🚀游戏逻辑设置

前面的东西聊完了,那么我们就不得不聊聊我们的游戏逻辑设计,我们得先想明白我们下这个三子棋需要什么,棋盘,棋子对不对,这个至少不能少吧,那下面呢,是不是得看怎样才算赢了这了这盘棋,接着还需要考虑我们和谁下棋得问题,当然小赵在这里剧透一波,这里是和电脑哦!怎么样是不是瞬间对这个代码感到好奇了,那我们抓紧开始吧。

⚔三子棋棋盘设计

首先是棋盘设置,在这里小赵找到了一张我们正常的井字棋的图片。

小赵仿照这个图做了一个我们可以通过编程实现的较为简单的棋盘

​​​​​​​

我们可以注意到这样的一个棋盘就更好实现,因为里面的很多元素是重复的而且可以靠我们|-——来解决。当然这里我们还是和扫雷时候一样进行一个封装,将这个代码装入我们的函数中。

#define ROW  3//定义行
#define COL  3//定义列
void InitBoard(char arr[ROW][COL], int row, int col)
{
	for (int i = 0; i < row; i++)
	{
		for (int j = 0; j < col; j++)
		{
			arr[i][j] = ' ';//对数组初始化,每个位置都是空白
		}
	}
}
void PrintBoard(char arr[ROW][COL], int row, int col)
{
	for (int i = 0; i < row; i++)
	{
		for (int j = 0; j < col; j++)
		{
			printf(" %c ", arr[i][j]);//打印占棋子位子
			if (j < col - 1)
			   printf("|");//打印棋盘元素//   |     |    (最后一个没有)
		}
		printf("\n");//换行
	for (int j = 0; j < col; j++)
	{
		printf("---");
		if (j < col - 1)
		printf("|");//———|———|———(最后一个没有|)
	}
	printf("\n");
	}
}

那么我们的整个棋盘就算结束了,到这里。 

🚀下棋

下面就是下棋了,在这里我们先约定一下我们的棋子是什么,在这里小赵就定义一下*为我们的棋子,那么我们每次下棋实际上就是将这些空白地方放上我们的棋子,也就是取代我们的%c的空白位置。

👀怎么设置人下棋

那么人怎么下棋呢,你完全可以根据小赵的上周的博客那篇教你学C里面讲的二维数组一样,把这个棋盘的下棋位置当成二维数组的每个位置,那么我们在下的时候只需要根据我们的每个坐标来下棋就好了,在这里小赵也是封装成了一个函数。

void PersonSet(char arr[ROW][COL], int row, int col)
{
	   int ret;
	     do
		{
			int x, y;
			scanf("%d %d", &x, &y);
			if (x-1 < row && y-1 < col)
			{
				if (arr[y-1][x-1] == ' ')
				{
					arr[y-1][x-1] = '*';//对第几列第几行下棋
					ret = 0;
				}
			}
			else
			{
				printf("输入错误,请重新输入\n");//报错重新输入
				ret = 1;
			}
		 } while (ret);
}

为了方便大家知道自己的棋是怎么下的,小赵又画了一张图供大家参考,当然画的不好请见谅。

👀怎么设置电脑下棋

这里的电脑下棋,小赵实在也不知道该如何去做一个去能分析对手下棋的强大AL,所以很抱歉,这里小赵用的还是未知数,当然如果大家有什么好的想法可以在小赵的评论区留下留言,让小赵也能学习学习。好了我们接着看电脑下棋,这里为了有所区别,小赵这里给电脑另一个棋字#,让我们有所区分。这里也是封装成一个函数了。

void ComputerSet(char arr[ROW][COL], int row, int col)
{
	srand((unsigned int)time(NULL));
	while (1) 
	{
		int x = rand() % row + 1;
		int y = rand() % col + 1;

		if (arr[y - 1][x - 1] == ' ')
		{
			arr[y - 1][x - 1] = '#';
			break;

		}
	}
}

🚀如何判断输赢

那么这个时候就是最关键的时候了。我们该如何判断输赢了,我们知道井字棋的结局共有三种,第一种是人赢了,第二种电脑赢,第三种平局(即棋盘下满了)。而且还要考虑到什么情况横着竖着和斜着,这里可以说是相当复杂了。我们一步一步来。

✊平局

平局怎么判定,其实就是棋盘上的棋子都下满了,那我们只需要用循环一个个扫就行了,确保棋盘上没有一个空的地方,且这个时候双方没有输赢(这个放在下面)。

int is_FULL(char arr[ROW][COL], int row, int col)
{
	for (int i = 0; i < row; i++)
	{
		for(int j=0;j<col;j++)
			if (arr[i][j] =' ')
			{
				return 0;//如果有空位,返回零
			}
	}
		return 1;//扫完所有放棋子,都没有空位,返回1
}

这里小赵用一种返还数字的方式来判断究竟有木有空的地方,这样我们在下面只需要将判断的三种结局的方式用一个函数穿起来就好了。 

✊判断输赢

在判断输赢这里,我们要尽可能的巧妙和省代码,比如判断人赢还是电脑赢,其实就是判断某一种棋子是否连三个相等。其次就是我们上面的平局用的1/0我们可以加入其中,如若是平局就进入if函数来出来,同时我们还要注意我们这个是一个判断,那么这个判断的时候就要注意我们判断的时候可能不是输赢平,那我们就要继续下,那我们可以再设置一个返还数字在下面,这样经过所有判断都没结果后,我们就输出一个字母意思是继续。

char is_win(char arr[ROW][COL], int row, int col)
{
	int i = 0;
	for (i = 0; i < row; i++)
	{
		if (arr[i][0] ==arr[i][1] && arr[i][1] == arr[i][2] && arr[i][0] != ' ')
		{
			return arr[i][0];//如果一竖行都是一种棋子,则返回这个棋子
		}
	}
	for (i = 0; i < col; i++)
	{
		if (arr[0][i] == arr[1][i] && arr[1][i] == arr[2][i] && arr[0][i] != ' ')
		{
			return arr[0][i];//如果一横行都是一种棋子,则返回这个棋子
		}
	}

	if (arr[0][0] == arr[1][1] && arr[1][1] ==arr[2][2] && arr[1][1] != ' ')
	{
		return arr[1][1];//如果左斜着都是一种棋子,则返回这个棋子
	}

	if (arr[0][2] == arr[1][1] &&arr[1][1] == arr[2][0] && arr[1][1] != ' ')
	{
		return arr[1][1];//如果右斜着都是一种棋子,则返回这个棋子
	}

	//判断平局
	if (is_FULL(arr, row, col))//返回1为真,0为假。
	{
		return 'Q';//如果是平局返还Q
	}

	return 'C';//若上面都不是,返还c意思是游戏继续

}

🛸游戏运行逻辑

我们现在手上已经有了这么多的函数下面就要编我们的整个运行的逻辑吧。首先我们要确定的是我们的判断应该什么时候进行,我觉得应该下一次就判断一次,因为我们的函数中是有继续的选项的,所以完全可以下一次判断一次,同时下一次打印一次。

void game()
{
	char ret;
	char arr[ROW][COL];
	InitBoard(arr, ROW, COL);//定义数组
	PrintBoard(arr, ROW, COL);//打印棋盘
	while (1)
	{
		PersonSet(arr, ROW, COL);//人下
		PrintBoard(arr, ROW, COL);//打印棋盘
		 ret = is_win(arr, ROW, COL);//判断输赢
		if (ret != 'C')//如果输出的不是c,就是不是继续,跳出循环
		{
			
			break;
		}
		printf("电脑下\n");
		ComputerSet(arr, ROW, COL);//电脑下
		PrintBoard(arr, ROW, COL);//打印棋盘
		ret = is_win(arr, ROW, COL);//判断
		if (ret != 'C')
		{
			
			break;
		}
	}
		if (ret == '*')//输出人的棋
		{
			printf("玩家赢\n");
		}
		else if (ret == '#')//输出电脑的棋
		{
			printf("电脑赢\n");
		}
		else//都不是,即使Q
		{
			printf("平局\n");
		}
	
	}

🌍整个代码

最后我们将我们的所有代码像之前的扫雷一样,装在不同的文件中。

game.h

#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<time.h>
#pragma once
#define ROW  3
#define COL  3
void InitBoard(char arr[ROW][COL], int row, int col);
void PrintBoard(char arr[ROW][COL], int row, int col);
void PersonSet(char arr[ROW][COL], int row, int col);
void ComputerSet(char arr[ROW][COL], int row, int col);
char is_win(char arr[ROW][COL], int row, int col);

game.c

#define _CRT_SECURE_NO_WARNINGS 1
#include"game.h"
void meun()
{
	printf("*************************************************\n");
	printf("*********************1.exit**********************\n");
	printf("*********************2.play**********************\n");
	printf("*************************************************\n");
	printf("请输入你的选择:");
}
void game()
{
	char ret;
	char arr[ROW][COL];
	InitBoard(arr, ROW, COL);
	PrintBoard(arr, ROW, COL);
	while (1)
	{
		PersonSet(arr, ROW, COL);
		PrintBoard(arr, ROW, COL);
		 ret = is_win(arr, ROW, COL);
		if (ret != 'C')
		{
			
			break;
		}
		printf("电脑下\n");
		ComputerSet(arr, ROW, COL);
		PrintBoard(arr, ROW, COL);
		ret = is_win(arr, ROW, COL);
		if (ret != 'C')
		{
			
			break;
		}
	}
		if (ret == '*')
		{
			printf("玩家赢\n");
		}
		else if (ret == '#')
		{
			printf("电脑赢\n");
		}
		else
		{
			printf("平局\n");
		}
	
	}

int main()
{
	int a = 0;
	do
	{
		meun();
		scanf("%d", &a);
		switch (a)
		{
		case 2:
			game();
			break;
		case 1:
			printf("游戏结束\n");
			break;
		default:
			printf("输入错误,请重新输入\n");
			break;
		}
	} while (a);
}

✍结语

好了今天的分享就到这里了,大家等会可以试着去敲属于自己的三子棋C语言游戏哦!如果大家喜欢小赵的C语言游戏系列,也可以订阅小赵的C语言小游戏系列,如果大家对这个游戏代码有更好的改进完善方案也可以在评论区留言哦,小赵会一一回复的。

如果觉得文章对你有帮助的话,还请点赞,关注,收藏支持小赵,如有不足还请指点,小赵及时改正,感谢大家支持!!!

​​​​​​​​​​​​​​

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

ArcGIS平滑处理栅格数据

一、实验背景 基于栅格数据的空间分析&#xff0c;常常需要根据特定的分析场景对栅格数据进行处理&#xff0c;如栅格数据的噪声处理。噪声是属性值具有突跃特征的像元位置&#xff0c;直接对带有噪声的栅格数据进行分析会对结果造成较大的影响。而降噪的主要方法之一是平滑&a…

12.4_黑马MybatisPlus笔记(下)

目录 11 12 thinking&#xff1a;关于Mybatis Plus中BaseMapper和IService&#xff1f; 13 ​编辑 thinking&#xff1a;CollUtil.isNotEmpty? 14 thinking&#xff1a;Collection、Collections、Collector、Collectors&#xff1f; thinking&#xff1a;groupBy&#…

风格迁移网络修改流程(自用版)

一. AdaAttN-Revisit Attention Mechanism in Arbitrary Neural Style Transfer&#xff08;ICCV2021&#xff09; 下载vgg_normalised.pth打开visdom python -m visdom.server在 train_adaattn.sh 中配置 content_path、style_path 和 image_encoder_path&#xff0c;分别表…

FFmpeg在Centos服务器上离线安装(包含所需依赖)并实现拉取rtsp流与推送至rtmp服务器

场景 Windows上使用FFmpeg实现rtsp视频流推流到RTMP流媒体服务器(EasyCVR流媒体服务器)&#xff1a; Windows上使用FFmpeg实现rtsp视频流推流到RTMP流媒体服务器(EasyCVR流媒体服务器)_rtsp 转流-CSDN博客 上面讲了在windows上ffmpeg的应用示例&#xff0c;如果是在centos服…

Hadoop进阶学习---HDFS分布式文件存储系统

1.hdfs分布式文件存储的特点 分布式存储:一次写入,多次读取 HDFS文件系统可存储超大文件,时效性较差. HDFS基友硬件故障检测和自动快速恢复功能. HDFS为数据存储提供很强的扩展能力. HDFS存储一般为一次写入,多次读取,只支持追加写入,不支持随机修改. HDFS可以在普通廉价的机器…

canvas绘制小丑

说明&#xff1a; 借鉴博主基于canvas绘制一个爱心(10行代码就够了) - 掘金 (juejin.cn) 代码实现 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta http-equiv"X-UA-Compatible" content&quo…

震坤行2023安全月活动顺利收官

震坤行2023安全月活动顺利收官 2023年6月&#xff0c;是第22个全国“安全生产月”&#xff0c;主题为 “人人讲安全、个个会应急”。震坤行工业超市(上海)有限公司基于国家 “安全生产月”的主题要求&#xff0c;以及公司具体的安全形势&#xff0c;于6月1日在全公司范围内正式…

EM32DX-E4【C#】

1外观&#xff1a; ecat总线&#xff0c;分布式io 2电源&#xff1a; 靠近SW拨码&#xff1a;24V 中间&#xff1a;0V 靠近面板&#xff1a;PE接地 3DI&#xff1a; 6000H DI输入寄存器 16-bit &#xff08;16位输入&#xff09; 00H U8 子索引总数 01H Unsigned16 IN1…

TCP 半连接队列和全连接队列

在 TCP 三次握手的时候&#xff0c;Linux 内核会维护两个队列&#xff0c;分别是&#xff1a; 半连接队列&#xff0c;也称 SYN 队列&#xff1b; 全连接队列&#xff0c;也称 accept 队列&#xff1b; 服务端收到客户端发起的 SYN 请求后&#xff0c;内核会把该连接存储到半连…

数据库系统概论期末经典大题讲解(用关系代数进行查询)

今天也是结束的最为密集的考试周&#xff0c;在分析过程中自己也有些许解题的感悟&#xff0c;在此分享出来&#xff0c;希望能帮到大家期末取得好成绩。 一.专门的关系运算 1.选择&#xff08;σ&#xff09; 选择操作符用于从关系中选择满足特定条件的元组 例如&#xff0c;…

计算机组成原理学习-总线总结

复习本章时&#xff0c;思考以下问题&#xff1a; 1)引入总线结构有什么好处&#xff1f;2)引入总线结构会导致什么问题&#xff1f;如何解决&#xff1f;

【专题】【中值定理-还原大法】

1&#xff09;构造辅助函数 2&#xff09;罗尔定理&#xff1a; 闭区间连续&#xff0c;开区间可导 F&#xff08;a&#xff09;F&#xff08;b&#xff09; 3&#xff09;F‘&#xff08;ξ&#xff09;0&#xff0c;原命题得证 极限保号性&#xff1a;

WPS论文写作——公式和公式序号格式化

首先新建一个表格&#xff0c;表格尺寸按你的需求来确定&#xff0c;直接 插入--》表格 即可。 然后在表格对应位置填上公式&#xff08;公式要用公式编辑器&#xff09;和公式序号&#xff0c;然后可以按照单独的单元格或者整行或整列等来设置样式&#xff0c;比如居中对齐、…

微服务实战系列之Cache(技巧篇)

前言 凡工具必带使用说明书&#xff0c;如不合理的使用&#xff0c;可能得到“意外收获”。这就好比每个人擅长的领域有所差异&#xff0c;如果放错了位置或用错了人&#xff0c;也一定会让 Leader 们陷入两难之地&#xff1a;“上无法肩负领导之重托&#xff0c;下难免失去伙伴…

123456前端调AES加密方法变为YehdBPev

使用密码加密服务: pig4cloud 加密服务

(华为)网络工程师教程笔记(网工教程)网工入门——3、静态路由路由表的配置

参考文章&#xff1a;【全236集】网络工程师从基础入门到进阶必学教程&#xff01;通俗易懂&#xff0c;2023最新版&#xff0c;学完即可就业&#xff01;网工入门_华为认证_HCIA_HCIP_数据通信_网工学习路线 文章目录 13. 网工入门10-静态路由&#xff08;路由表的配置&#x…

传输层可靠传输的原理

目录 1.停止等待协议 2.连续ARQ协议 3.TCP报文段的首部格式 4.TCP的滑动窗口机制 &#xff08;1&#xff09;发送窗口 &#xff08;2&#xff09;接收窗口 &#xff08;3&#xff09;发送缓存 5.超时重传时间的选择 6.选择确认SACK(Selective ACK) 7.使用滑动窗口实现…

TCP三次握手与四次挥手:推荐学习资料、过程详解、面试相关题与回答模板(为什么不是两次握手等精讲)

推荐资料&#xff08;建议按照顺序先都看完&#xff0c;再看本篇文章&#xff09; https://www.bilibili.com/video/BV18h41187Ep/ https://www.bilibili.com/video/BV1at4y1Q77b/ https://bbs.huaweicloud.com/blogs/277728 https://blog.csdn.net/dreamispossible/article/d…

vm net 方式 静态ip配置访问主机IP和外网

1、win 11 安装vm&#xff0c;镜像文件 F:\software\VMwork\CentOS-7-x86_64-Everything-1804.iso 2、配置网络 net 方式 3、右击网络--》属性---》更改适配器设置--》vmnet8 属性、这里不做配置会出现主机ping通访问不通的情况&#xff0c;&#xff08;访问不通&#xff0c;…

pytorch 数据预加载

1. Abstract 本文介绍一个工具 PreDataLoader&#xff0c;它包装 torch.utils.data.DataLoader&#xff0c;接收该类的一个实例 loader&#xff0c;启动一个线程 t&#xff0c;创建一个队列 q&#xff0c;t 将 loader 中的数据预加载到队列 q 中, 以在模型计算时也能启动启动数…