C语言初阶之三子棋

news2024/11/26 0:34:24

在这里插入图片描述

三子棋

  • 初始文件建立
  • 主函数文件
  • 棋盘的初始化(InitBoard)
  • 打印棋盘(DisplayBoard)
  • 玩家下棋(PlayerMove)
  • 电脑下棋(ComputerMove)
  • 判定胜负
  • 所有代码
    • game.h
    • game.c
    • test.c
  • 结语

初始文件建立

首先我们建立game.h,game.c,test.c三个文件,如下图
在这里插入图片描述
在game.h文件中我们引用需要使用到的头文件和定义标识符以及函数定义,其余两个文件包含game.h文件即可,即#include"game.h"。
game.h代码如下:

#pragma once
#include<stdio.h>
#include<stdlib.h>
#include<time.h>
#define ROW 3
#define COL 3

void InitBoard(char board[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]);//判断胜负

其中ROW和COL两个标识符定义的是棋盘的行和列大小,方便我们在写函数和使用是重复使用数字,有更高的辨识度,表达更清晰。

主函数文件

代码如下:

#define _CRT_SECURE_NO_WARNINGS 1
#include"game.h"
void menu()
{
	printf("*****************************************\n");
	printf("********   1.play       0.exit   ********\n");
	printf("*****************************************\n");
}
void game()
{
	char board[ROW][COL] = { 0 };
	InitBoard(board);
	DisplayBoard(board);
	char ret = 0;
	while (1)
	{
		PlayerMove(board);
		DisplayBoard(board);
		ret = IsWin(board);
		if (ret != 'c')
			break;
		printf("电脑下:\n");
		ComputerMove(board);
		DisplayBoard(board);
		ret = IsWin(board);
		if (ret != 'c')
			break;
	}
	if (ret == 'O')
		printf("玩家赢!!!\n");
	else if(ret == 'X')
		printf("电脑赢!!!\n");
	else
		printf("平局\n");
}
int main()
{
	int input = 0;
	srand((unsigned int)time(NULL));
	do
	{
		menu();
		printf("请选择:>");
		scanf("%d", &input);
		switch (input)
		{
		case 1:
			game();
			break;
		case 0:
			printf("退出游戏\n");
			break;
		default:
			printf("选择错误,重新选择!\n");
			break;
		}
	} while (input);
	return 0;
}

具体思路:
首先我们从主函数看,建立整形input是为了后面让玩家选择游戏选项,然后就是随机数的时间种子建立,在后面的电脑下棋会用到,下面主体就是这个do…while循环,首先是menu菜单,这个函数里包含的就是游戏选项的打印,然后下面的scanf的用处就是为了记录玩家的游戏选项的输入,再由switch语句进行语句判定,输入1进入游戏,输入0退出游戏,输入其他则退出游戏,我们再来看game函数,首先建立棋盘(其本质是一个二维数组),值默认为0,需要注意的是,这里我们下棋时,棋盘内放什么,棋盘类型就定义什么,比如这里我放的是字符,那就使用char类型;再进行初始化,初始化后数组内均为空格,然后打印棋盘,建立一个字符类型ret,用于接收判断胜负的信息,然后就是关于人机下棋的一个循环,只要没有胜负结果跳出,就一直循环;然后就是玩家下棋,再打印,此时有一个判断,这里的’c’再判断胜负函数中是继续的信息,也就是说,如果不满足继续下的条件,就跳出,电脑下同理,最后一段就是胜负的判定,如果接收的返回值是’O’,那么玩家胜利,如果接收的返回值是’X’,那么电脑胜利,否则就是平局,具体实现看下面的函数搭建

棋盘的初始化(InitBoard)

从这里开始,下面所有的函数都是包含在game.c文件中的
代码如下:

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

具体思路:
这里就是利用两个循环将棋盘中所有元素初始化为空格

打印棋盘(DisplayBoard)

代码如下:

void DisplayBoard(char board[ROW][COL])
{
	int i = 0;
	for (i = 0; i < ROW; i++)
	{
		int j = 0;
		for (j = 0; j < COL; j++)
		{
			printf(" %c ", board[i][j]);
			if (j < ROW - 1)
				printf("|");
		}
		printf("\n");
		if(i < ROW - 1)
		{
			for (j = 0; j < COL; j++)
			{
				printf("---");
				if (j < COL - 1)
					printf("|");
			}
			printf("\n");
		}
	}
}

具体思路:
首先我们看下面这个图,即为要打印棋盘的效果
在这里插入图片描述
我们把棋盘分为两种类型,按行打印,上面这种写法可以在需要时改变棋盘大小,而不再需要改变函数,首先我们看第一行包含空格棋子空格竖杠,总共是3个,最后一个没有竖杠,由此第一行类型的打印循环就写出来了,再看第二行包含三个横杠加一个竖杠,最后一个不加,还需要注意的是,最后一行不打印这个,因此第二行类型打印循环也写出,整个打印循环原理就是这样

玩家下棋(PlayerMove)

代码如下:

void PlayerMove(char board[ROW][COL])
{
	int x = 0, y = 0;
	printf("玩家下棋:>\n");
	while (1)
	{
		printf("请输入下棋坐标,中间用空格隔开:>");
		scanf("%d%d", &x, &y);
		if(x>=1&&x<=3&&y>=1&&y<=3)
		{
			if (board[x - 1][y - 1] == ' ')
			{
				board[x - 1][y - 1] = 'O';
				break;
			}
			else
				printf("此处被占用,请重新落子\n");
		}
		else
		{
			printf("坐标非法,请重新落子\n");
		}
	}
}

具体思路:
首先我们创建xy两个变量用于接收玩家要落子的坐标,这里的while循环是为了防止玩家输错坐标而建立的,如果输入正确就会跳出循环,这里的判定条件是以行列为1-3的标准,所以在二维数组中还要减去1才能找到正确位置,下面的判定条件是为空才能落子,被占用或者输入非法坐标都会重新进入循环再输入坐标。玩家落子为字母O。
如图:
在这里插入图片描述

电脑下棋(ComputerMove)

代码如下:

void ComputerMove(char board[ROW][COL])
{

	while (1)
	{
		int x = rand()%ROW, y = rand()%COL;
			if (board[x][y] == ' ')
			{
				board[x][y] = 'X';
				break;
			}
	}
}

具体思路:
这里就用到了这个以时间为种子的随机数,将它模上行和列,得到的即为0-2的随机值,判定条件也是不为空就落子。电脑落子为字母X。
如图:
在这里插入图片描述

判定胜负

代码如下:

char IsWin(char board[ROW][COL])
{
	int i = 0;
	for (i = 0; i < ROW; i++)
	{
		if (board[i][0] == board[i][1] && board[i][1] == board[i][2] && board[i][0] != ' ')//行
		{
			return board[i][0];
		}
	}
	for (i = 0; i < COL; i++)
	{
		if (board[0][i] == board[1][i] && board[1][i] == board[2][i] && board[0][i] != ' ')//列
		{
			return board[0][i];
		}
	}
	if (board[0][0] == board[1][1] && board[1][1] == board[2][2] && board[0][0] != ' ')//对边
		return board[0][0];
	if (board[0][2] == board[1][1] && board[1][1] == board[2][0] && board[0][2] != ' ')//对边
		return board[0][2];
	if (IsFull(board) == 1)
		return 'q';
	return 'c';
}

具体思路:
三子棋的胜负只有三种情况,就是同行相同、同列相同、对角相同,根据这几种情况写出判定想等且不为空的条件语句,然后返回任意一个条件中任意一个元素即可。最后判定是不是满了,如果棋盘满了返回任意值都可(不等于’c’就行,'c’是继续的条件),最后上面的条件都不满足,则继续。
这里还有一个判定棋盘是否满了的函数
代码如下:

int IsFull(char board[ROW][COL])
{
	int i = 0,j = 0;
	for (i = 0; i < ROW; i++)
	{
		for (j = 0; j < COL; j++)
		{
			if (board[i][j] == ' ')
				return 0;
		}
	}
	return 1;
}

就是对二维数组的遍历匹配,如果有空位置,则返回0,都不符合,则返回1。
胜负判定如图:
平局
在这里插入图片描述
玩家赢
在这里插入图片描述
电脑赢
在这里插入图片描述
因为随机数的原因,其实要赢很简单,但是要输很难,所以这段代码实现游戏的最终目的是让电脑赢(手动滑稽)

所有代码

game.h

#pragma once
#include<stdio.h>
#include<stdlib.h>
#include<time.h>
#define ROW 3
#define COL 3

void InitBoard(char board[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]);

game.c

#define _CRT_SECURE_NO_WARNINGS 1
#include"game.h"
void InitBoard(char board[ROW][COL])
{
	int i = 0;
	for (i = 0; i < ROW; i++)
	{
		int j = 0;
		for (j = 0; j < COL; j++)
		{
			board[i][j] = ' ';
		}
	}
}
void DisplayBoard(char board[ROW][COL])
{
	int i = 0;
	for (i = 0; i < ROW; i++)
	{
		int j = 0;
		for (j = 0; j < COL; j++)
		{
			printf(" %c ", board[i][j]);
			if (j < ROW - 1)
				printf("|");
		}
		printf("\n");
		if(i < ROW - 1)
		{
			for (j = 0; j < COL; j++)
			{
				printf("---");
				if (j < COL - 1)
					printf("|");
			}
			printf("\n");
		}
	}
}
void PlayerMove(char board[ROW][COL])
{
	int x = 0, y = 0;
	printf("玩家下棋:>\n");
	while (1)
	{
		printf("请输入下棋坐标,中间用空格隔开:>");
		scanf("%d%d", &x, &y);
		if(x>=1&&x<=3&&y>=1&&y<=3)
		{
			if (board[x - 1][y - 1] == ' ')
			{
				board[x - 1][y - 1] = 'O';
				break;
			}
			else
				printf("此处被占用,请重新落子\n");
		}
		else
		{
			printf("坐标非法,请重新落子\n");
		}
	}
}
void ComputerMove(char board[ROW][COL])
{

	while (1)
	{
		int x = rand()%ROW, y = rand()%COL;
			if (board[x][y] == ' ')
			{
				board[x][y] = 'X';
				break;
			}
	}
}

int IsFull(char board[ROW][COL])
{
	int i = 0,j = 0;
	for (i = 0; i < ROW; i++)
	{
		for (j = 0; j < COL; j++)
		{
			if (board[i][j] == ' ')
				return 0;
		}
	}
	return 1;
}


char IsWin(char board[ROW][COL])
{
	int i = 0;
	for (i = 0; i < ROW; i++)
	{
		if (board[i][0] == board[i][1] && board[i][1] == board[i][2] && board[i][0] != ' ')
		{
			return board[i][0];
		}
	}
	for (i = 0; i < COL; i++)
	{
		if (board[0][i] == board[1][i] && board[1][i] == board[2][i] && board[0][i] != ' ')
		{
			return board[0][i];
		}
	}
	if (board[0][0] == board[1][1] && board[1][1] == board[2][2] && board[0][0] != ' ')
		return board[0][0];
	if (board[0][2] == board[1][1] && board[1][1] == board[2][0] && board[0][2] != ' ')
		return board[0][2];
	if (IsFull(board) == 1)
		return 'q';
	return 'c';
}

test.c

#define _CRT_SECURE_NO_WARNINGS 1
#include"game.h"
void menu()
{
	printf("*****************************************\n");
	printf("********   1.play       0.exit   ********\n");
	printf("*****************************************\n");
}
void game()
{
	char board[ROW][COL] = { 0 };
	InitBoard(board);
	DisplayBoard(board);
	char ret = 0;
	while (1)
	{
		PlayerMove(board);
		DisplayBoard(board);
		ret = IsWin(board);
		if (ret != 'c')
			break;
		printf("电脑下:\n");
		ComputerMove(board);
		DisplayBoard(board);
		ret = IsWin(board);
		if (ret != 'c')
			break;
	}
	if (ret == 'O')
		printf("玩家赢!!!\n");
	else if(ret == 'X')
		printf("电脑赢!!!\n");
	else
		printf("平局\n");
}
int main()
{
	int input = 0;
	srand((unsigned int)time(NULL));
	do
	{
		menu();
		printf("请选择:>");
		scanf("%d", &input);
		switch (input)
		{
		case 1:
			game();
			break;
		case 0:
			printf("退出游戏\n");
			break;
		default:
			printf("选择错误,重新选择!\n");
			break;
		}
	} while (input);
	return 0;
}

结语

近期更新大量C语言的博客,希望大家多支持!!!

有兴趣的小伙伴可以关注作者,如果觉得内容不错,请给个一键三连吧,蟹蟹你哟!!!
制作不易,如有不正之处敬请指出
感谢大家的来访,UU们的观看是我坚持下去的动力
在时间的催化剂下,让我们彼此都成为更优秀的人吧!!!
在这里插入图片描述

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

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

相关文章

一文总结Redis面试知识点

文章目录 Redis知识点知识点1 缓存穿透知识点2 布隆过滤器知识点3 缓存击穿知识点4 缓存雪崩知识点4 双写一致性知识点5 Redis持久化知识点6 数据过期策略知识点7 数据淘汰策略知识点8 分布式锁知识点9 主从复制知识点10 哨兵模式知识点11 分片集群知识点12 IO多路复用 Redis问…

零信任安全:综合网络安全方法

在当今的数字世界&#xff0c;网络威胁日益复杂&#xff0c;使得组织更加难以保护敏感数据和系统的安全。为应对当前的网络威胁&#xff0c;全新的“零信任”安全模型应运而生。 零信任安全作为一种网络安全方法&#xff0c;要求组织必须验证每一次访问尝试&#xff0c;无论访…

云发布的Docker部署文档

云发布的Docker部署文档 文章目录 云发布的Docker部署文档1、服务器安装 Docker2、修改后端微服务的配置2.1 修改 MySQL 的配置2.2 修改 Redis 的配置2.3 修改 Nacos 的配置 3、生成微服务镜像4、拉取远程镜像5、生成前端镜像5.1 准备文件5.2 构建前端镜像 6、基于镜像生成并运…

《编程思维与实践》1062.计算2的N次方

《编程思维与实践》1062.计算2的N次方 题目 思路 法一:直接复用计算a的n次方的代码即可. 法二:注意到 2 30 2^{30} 230在int(32位,第一位为符号位, 2 31 − 1 2^{31}-1 231−1)的范围内,可以直接处理. 注: int范围为 [ − 2 31 , 2 31 − 1 ] [-2^{31},2^{31}-1] [−231,231−…

利用Eigen求解不同坐标系的旋转矩阵

不同坐标系之间的刚性转换以及实现&#xff1a; 坐标系转换原理如下&#xff0c;使用了仿射变换实现了旋转平移&#xff1a; [ r 11 r 12 t x r 21 r 22 t y 0 0 ] ∗ [ x y 1 ] [ x ′ y ′ 1 ] \begin{bmatrix} r_{11} & r_{12}& t_x\\ r_{21}& r_{22}& t…

有没有免费的云渲染平台?

有没有免费的云渲染平台&#xff1f;相信很多人在找云渲染平台的时候都会先网上查找下有没有免费的平台可以使用&#xff0c;而且也确实看到很多云渲染平台写着“免费渲染”&#xff0c;所以这免费是真的免费还是只是一个噱头呢&#xff1f;下面&#xff0c;由前沿的 云渲染平台…

如何应对洗钱犯罪的新变化?探索图数据库在反洗钱中的应用

背景导读 洗钱&#xff0c;即对非法资金的合法化过程。它由违法犯罪集团组织&#xff0c;在躲避银行和执法部门监测情况下&#xff0c;利用违法交易手段对非法收益进行转移&#xff0c;是金融犯罪领域最难发现的活动之一。当前&#xff0c;洗钱团伙日趋集团化、规模化和专业化…

服务熔断Hystrix入门

目录 一. 微服务的核心知识1.1 雪崩效应1.2 服务隔离1.3 熔断降级1.4 服务限流 二. Hystrix介绍三. Hystrix服务熔断的实现方式3.1 Rest 实现熔断3.2 Feign实现服务熔断 四.服务熔断Hystrix高级4.1 Hystrix的监控平台4.2 搭建Hystrix DashBoard监控4.2.1 熔断器的状态4.2.1.1 测…

成为一名高级网络安全工程师,你需要学习什么?

前言 这是我的建议如何成为网络安全工程师&#xff0c;你应该按照下面顺序学习。 简要说明 第一件事你应该学习如何编程&#xff0c;我建议首先学python&#xff0c;然后是java。 &#xff08;非必须&#xff09;接下来学习一些算法和数据结构是很有帮助的&#xff0c;它将帮…

ChatGpt vs 文心一言,谁更厉害

前言 ​ 当下最火的人工智能毫无疑问是大模型的聊天机器人&#xff0c;ChatGpt的出现&#xff0c;让无数人为之惊叹&#xff0c;它聪明、逻辑清晰、又什么都懂&#xff0c;甚至被人们认为是第四次工业革命的开端。AI的大模型之争&#xff0c;美国的OpenAI已经率先拔得头筹&…

Path如何进行环境变量的配置?

开发Java程序&#xff0c;需要使用JDK提供的开发工具(比如javac.exe、java.exe等命令)&#xff0c;而这些工具在JDK的安装目录的 bin目录下&#xff0c;如果不配置环境变量&#xff0c;那么这些命令只可以在该目录下执行。我们不可能把所有的java文件都放到JDK 的bin目录下&…

笛卡尔坐标系和Frenet坐标系

笛卡尔坐标系和Frenet坐标系 Frenet坐标系&#xff0c;自动驾驶“混乱”的源头 【自动驾驶】运动规划丨轨迹规划丨Frenet坐标转换 【自动驾驶】笛卡尔坐标系和frenet坐标系相互转换 轨迹规划作为具有多个非线性约束的高维优化问题&#xff0c;需要消耗大量的计算资源&#xff0…

前端012_标签模块_条件查询实现

标签模块_条件查询实现 1、需求分析2、Mock添加分类数据3、Api调用分类接口4、添加条件查询模版1、需求分析 在列表上方添加通过标签名称、分类名称的查询功能。 Form 表单参考 :https://element.eleme.cn/#/zh-CN/component/form#xing-nei-biao-danSelect选择器:https://el…

4-表约束和表设计

目录 1.表约束 1.1.什么是表约束&#xff1f; 1.2.为什么需要表约束&#xff1f; 1.3.常见的表约束 1.3.1.非空约束 not null --->a.查看非空约束 --->b.验证非空约束 1.3.2.唯一约束 unique --->a.查看唯一约束 --->b.验证唯一约束 --->PS&#xff…

error处理:org.springframework.web.bind.MissingServletRequestParameterException

在工作中遇到错误org.springframework.web.bind.MissingServletRequestParameterException 此时为POST方法&#xff0c;添加了RequestParam注解&#xff0c;如图 postman中请求如下 解决办法:选择form-data进行传参即可

linux 设备树详解

设备树 描述设备树的文件叫做 DTS(Device Tree Source)&#xff0c;这个 DTS 文件采用树形结构描述板级设备&#xff0c;也就是开发板上的设备信息&#xff0c;比如CPU 数量、 内存基地址、IIC 接口上接了哪些设备、SPI 接口上接了哪些设备等等。 树的主干就是系统总线&#x…

【服务器数据恢复】Linux系统下OA+oracle的数据恢复案例

服务器数据恢复环境&#xff1a; 某公司一台服务器中组建一组raid5磁盘阵列&#xff1b; 上层操作系统为linux redhat&#xff0c;部署OA系统&#xff0c;后端数据库为oracle。 服务器故障&初检&#xff1a; raid5中有2块磁盘先后掉线&#xff0c;服务器崩溃。oracle已经不…

springboot使用ECharts、ECharts html中文乱码、直接引用CDN资源文件和引用本地资源文件哪个好

springboot使用ECharts、动态地引用版本 1.添加依赖2.创建图表引入ECharts文件方式直接引用CDN资源文件和引用本地资源文件哪个好 3.映射4.添加配置加载ECharts资源5.测试访问6.升级版本号问题7.ECharts html中文乱码 如果想在Spring Boot应用程序中使用ECharts&#xff0c;则可…

多目标检测:基于Yolo优化的多目标检测(附论文下载)

关注并星标 从此不迷路 计算机视觉研究院 公众号ID&#xff5c;ComputerVisionGzq 学习群&#xff5c;扫码在主页获取加入方式 计算机视觉研究院专栏 作者&#xff1a;Edison_G 为了解决目标检测任务中小目标检测精度低、误检、漏检率高等问题&#xff0c;有研究者提出了一种新…

前端实战项目:网易云静态页面——主页面右侧部分

文章目录 前言main部分结构布局用户登陆右侧列表header的封装歌手列表主播列表申请按钮 总代码 前言 项目持续更新中&#xff5e; 网易云静态页面——导航栏 网易云静态页面——轮播图 Flex布局详解 所用到文件及文件夹 header&#xff1a;是对某些标题样式的封装 main&…