C语言实现三子棋教学

news2024/12/28 20:29:05

本篇博客会教你如何使用C语言实现三子棋。主要包含以下步骤:

  1. 初始化棋盘。
  2. 打印棋盘。
  3. 玩家下棋。
  4. 电脑下棋。
  5. 判断输赢
    三子棋

0.预备工作

先定义一些符号,后面会用到。主要是棋盘的大小(行数+列数),以及棋子。

#define ROW 3
#define COL 3

#define PLAYER_PIECE 'O'
#define COMPUTER_PIECE 'X'

1.初始化棋盘

棋盘是一个二维数组:

char board[ROW][COL] = { 0 };

棋盘一开始应该初始化为全空格,这样在对应的位置才会空出来。由于二维数组在内存中是连续存放的,可以使用memset进行初始化。

void InitBoard(char board[ROW][COL], int row, int col)
{
	memset(&board[0][0], ' ', row * col);
}

2.打印棋盘

棋盘的打印可以加上分隔符,把行列分隔开来,像这样:

void DisplayBoard(char board[ROW][COL], int row, int 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");
		}
	}
}

这是打印出来的效果:
打印效果

3.玩家下棋

玩家下棋时,需要分别判断该位置是否是合法范围,该位置是否被占用,然后再下棋:

void PlayerMove(char board[ROW][COL], int row, int col)
{
	int x = 0;
	int y = 0;

	printf("玩家下棋\n");
	while (1)
	{
		printf("请输入坐标:>");
		scanf("%d %d", &x, &y);
		if (x<1 || x>row || y<1 || y>col)
		{
			printf("坐标非法,请重新输入!\n");
		}
		else if (board[x - 1][y - 1] != ' ')
		{
			printf("改坐标已被占用,请重新输入!\n");
		}
		else
		{
			board[x - 1][y - 1] = PLAYER_PIECE;
			break;
		}
	}
}

4.电脑下棋

电脑下棋暂且选用随机下棋的方案。随机生成一个坐标,如果未被占用,就在该位置下棋。

void ComputerMove(char board[ROW][COL], int row, int col)
{
	printf("电脑下棋\n");
	while (1)
	{
		int x = rand() % row;
		int y = rand() % col;
		if (board[x][y] == ' ')
		{
			printf("坐标:>%d %d\n", x + 1, y + 1);
			board[x][y] = COMPUTER_PIECE;
			break;
		}
	}
}

5.判断输赢

设计一个函数来判断当前游戏的状态。IsWin函数会返回4种状态,分别是:

  • PLAYER_PIECE - 玩家胜利
  • COMPUTER_PIECE - 电脑胜利
  • Q - 平局
  • C - 未分胜负,继续游戏

判断输赢只需要判断每行、每列、每条对角线是否全部是某一方棋子。简化一下,就是判断某一行/列/对角线是否相等且不为空格。这可以写一个循环来判断。当然,要反向思维:如果遇到不相等的,或者遇到空格,则跳出循环。如果是因为循环条件判断为假而跳出循环,则说明都相等且没有遇到空格,返回任意位置就行了(因为玩家赢返回玩家棋子,电脑赢返回电脑棋子)。

判断平局也很简单,如果棋盘都满了,且没有分出胜负,就平局了。

static bool IsFull(char board[ROW][COL], int row, int col)
{
	for (int i = 0; i < row; ++i)
	{
		for (int j = 0; j < col; ++j)
		{
			if (board[i][j] == ' ')
			{
				return 0; // 没满
			}
		}
	}

	return 1; // 满了
}

char IsWin(char board[ROW][COL], int row, int col)
{
	int i = 0;
	int j = 0;

	// 行
	for (i = 0; i < row; ++i)
	{
		for (j = 1; j < col; ++j)
		{
			// 遇到空格或者和左边的不相等
			if (board[i][j] == ' ' || board[i][j] != board[i][j - 1])
			{
				break;
			}
		}
		if (j == col)
		{
			return board[i][0];
		}
	}

	// 列
	for (j = 0; j < col; ++j)
	{
		for (i = 1; i < row; ++i)
		{
			// 遇到空格或者和上边的不相等
			if (board[i][j] == ' ' || board[i][j] != board[i - 1][j])
			{
				break;
			}
		}
		if (i == row)
		{
			return board[0][j];
		}
	}

	// 左上->右下对角线
	for (i = 1; i < row; ++i)
	{
		// 遇到空格或者和左上的不相等
		if (board[i][i] == ' ' || board[i][i] != board[i - 1][i - 1])
		{
			break;
		}
	}
	if (i == row)
	{
		return board[0][0];
	}

	// 右上->左下对角线
	for (i = 1; i < row; ++i)
	{
		// 遇到空格或者和右上的不相等
		if (board[i][col - i - 1] == ' ' || board[i][col - i - 1] != board[i - 1][col - i])
		{
			break;
		}
	}
	if (i == row)
	{
		return board[0][col - 1];
	}

	// 判断平局
	if (IsFull(board, row, col))
	{
		return 'Q';
	}

	// 游戏继续
	return 'C';
}

测试游戏逻辑

void Menu()
{
	printf("************************\n");
	printf("***** 1. play      *****\n");
	printf("***** 0. exit      *****\n");
	printf("************************\n");
}

void Game()
{
	char board[ROW][COL] = { 0 };
	InitBoard(board, ROW, COL);
	DisplayBoard(board, ROW, COL);

	char ret = 0;
	int i = 0;
	// 轮流下棋
	while (1)
	{
		if (i++ % 2 == 0)
		{
			PlayerMove(board, ROW, COL);
		}
		else
		{
			ComputerMove(board, ROW, COL);
		}
		DisplayBoard(board, ROW, COL);
		ret = IsWin(board, ROW, COL);
		if (ret != 'C')
		{
			break;
		}
	}

	// 游戏结束
	if (ret == PLAYER_PIECE)
	{
		printf("恭喜你,你赢了!\n");
	}
	else if (ret == COMPUTER_PIECE)
	{
		printf("很遗憾,你输了!\n");
	}
	else
	{
		printf("平局,再接再厉!\n");
	}
}

void Test()
{
	srand((unsigned int)time(NULL));
	int input = 0;
	do
	{
		Menu();
		printf("请选择:>");
		scanf("%d", &input);
		switch (input)
		{
		case 1:
			Game();
			break;
		case 0:
			printf("退出游戏\n");
			break;
		default:
			printf("选择错误,请重新选择!\n");
			break;
		}
	} while (input);
}

int main()
{
	Test();

	return 0;
}

第一次测试 - 玩家胜利:
玩家胜利

第二次测试 - 电脑胜利:
电脑胜利

第三次测试 - 平局:
平局

总结

  1. 三子棋的实现需要先规划出步骤,因为C语言是面向过程的。
  2. 初始化棋盘利用了二维数组在内存中连续存储的特点。
  3. 打印棋盘是二维数组的遍历,在适当位置打印分隔符。
  4. 玩家下棋需要注意某些特殊情况的处理,比如坐标非法或者已占用。
  5. 电脑下棋需要生成随机数,详见我之前写的一篇博客。
  6. 判断输赢需要注意如何使用循环来判断都相等,方法是反向思维,如果不相等就跳出循环,如果没有跳出循环则说明都相等。

感谢大家的阅读!

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

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

相关文章

skvideo.io.vread无法读取视频(九天毕生版)

Vread无法读取视频 使用九天GPU时遇到的错误以及解决方法: 、vread无法读取视频 需要下载ffmpeg的exe&#xff08;从网上找&#xff09; 下载ffmpeg.exe&#xff08;一共三个&#xff09;后将exe的上级目录&#xff08;bin&#xff09;文件路径添加到系统路径中&#xff08;…

Grounding DINO-开集目标检测论文解读

文章目录摘要背景算法3.1Feature Extraction and Enhancer3.2. Language-Guided Query Selection3.3. Cross-Modality Decoder3.4. Sub-Sentence Level Text Feature3.5. Loss Function实验4.2 Zero-Shot Transfer of Grounding DINOCOCO数据集LVIS数据集ODinW&#xff0c;开放…

超级账本与区块链应用场景

文章目录 区块链3.0去中心化应用的新需求区块链技术在行业应用中的条件区块链3.0架构与超级账本 区块链3.0架构 超级账本(Hyperledger Fabric)超级账本的项目FabricFabric的典型运行模型在Fabric中完成一次交易的整体步骤Fabric的节点 链码(Chaincode)数字身份证书组织通道 区块…

Java之 重载 重写的区别

重载 在同一个类中&#xff0c;多个方法有相同的方法名&#xff0c;但参数列表不同&#xff0c;这种同名不同参的方法就是重载重写 子类在继承父类方法的基础上&#xff08;方法名和参数列表相同&#xff09;&#xff0c;对父类方法的实现进行覆盖的操作叫重写规则 重载的规则…

【详细教程】国内部署ChatGPT镜像网站

文章目录 一、准备阶段0、注册Open AI账号1、创建API密钥2、国内云服务器3、国外云服务器4、镜像网站代码5、效果重要&#xff1a;部署时会修改glibc库&#xff0c;为了防止云服务器被搞坏&#xff0c;请提前进行备份或者创建快照重要&#xff1a;部署时会修改glibc库&#xff…

Echarts 如何添加页脚元素

要在 Echarts 图表中添加页脚元素&#xff0c;可以通过在 Echarts 实例的配置对象中添加 graphic 元素来实现。graphic 元素是一个数组形式的配置项&#xff0c;可以通过其中的 text 元素添加文字&#xff0c;rect 元素添加矩形&#xff0c;image 元素添加图片等&#xff0c;从…

JavaSE注解

注解分类和说明点 注解&#xff1a;可对程序做解释可被其他程序读取 元注解&#xff1a;Target&#xff1a;表明注解的使用范围&#xff0c;Retention&#xff1a;表示要在什么级别保存注解信息&#xff0c;Document&#xff0c;Inherited 自定义注解&#xff1a;interface …

一文说透安全沙箱技术

在数字经济的东风中&#xff0c;数据安全至关重要。目前已经颁布了包括《数据安全法》、《个人信息保护法》和《数据安全管理办法》在内的国家政策&#xff0c;以促进整个数据要素的发展。 而近年来&#xff0c;随着移动应用程序的普及和小程序技术的崛起&#xff0c;安全沙箱…

RB-PEG-MAL,罗丹明-聚乙二醇-马来酰亚胺;Rhodamine PEG Maleimide;RB-PEG2000-MAL

RB-PEG-MAL,罗丹明-聚乙二醇-马来酰亚胺 中文名称&#xff1a;罗丹明-聚乙二醇-马来酰亚胺 英文名称&#xff1a;Rhodamine PEG Maleimide RB-PEG-MAL 性状&#xff1a;粉红色或暗红色固体或者液体,取决于分子量。 溶剂&#xff1a;溶于大部分有机溶剂,在水中有很好的溶解…

( “树” 之 BFS) 513. 找树左下角的值 ——【Leetcode每日一题】

513. 找树左下角的值 给定一个二叉树的 根节点 root&#xff0c;请找出该二叉树的 最底层 最左边 节点的值。 假设二叉树中至少有一个节点。 示例 1: 输入: root [2,1,3] 输出: 1 示例 2: 输入: [1,2,3,4,null,5,6,null,null,7] 输出: 7 提示: 二叉树的节点个数的范围是 […

2020年团体程序设计天梯赛-模拟赛

L1-5 判断题 判断题的评判很简单&#xff0c;本题就要求你写个简单的程序帮助老师判题并统计学生们判断题的得分。 输入格式&#xff1a; 输入在第一行给出两个不超过 100 的正整数 N 和 M&#xff0c;分别是学生人数和判断题数量。第二行给出 M 个不超过 5 的正整数&#xf…

ChatGPT和Claude 对比测试(以Review MLIR Codegen代码为例)

Claude在MLIR代码分析上完全超越了ChatGPT并表现十分惊艳&#xff0c;请阅读全文或者自己注册感受它的强大。 0x0. 前言 这里将以oneflow IR部分中的一个Codegen任务&#xff08;目标是在mlir codegen中支持oneflow stream&#xff0c;用oneflow stream替换pass中自己生成的st…

10个前端开发者需要掌握的DOM技巧

Web开发不断发展&#xff0c;掌握最新的趋势和最佳实践对每位开发者来说都至关重要。Web开发的最重要方面之一就是使用文档对象模型&#xff08;DOM&#xff09;。这篇文章中&#xff0c;小蓝将与大家共同探讨10个必须掌握的DOM技巧&#xff0c;帮助您成为更高效、更有效的开发…

CSS—javaEE

文章目录1.引入的方式1.1作为HTML的标签属性1.2style标签1.3link标签外部引入css文件2.基础选择器2.1标签选择器2.2类选择器2.3id选择器2.4通配符选择器3.复合选择器3.1后代选择器3.2子选择器3.3并集选择器3.4伪类选择器4.设置样式4.1字体4.2文本4.3背景4.4圆角5.显示模式6.盒模…

【Linux】HTTP超文本传输协议

http主要用在浏览器和web服务器通信&#xff0c;是基于TCP协议的应用层无连接的传输协议。 HTTP与HTTPS http是明文数据传输&#xff0c;如果攻击者截取了Web浏览器和网站服务器之间的传输报文&#xff0c;就可以直接读懂其中的信息。协议使用端口号&#xff1a;80。 https更…

AIGC:数字内容创新的新引擎,还有藏着更多你知道的细节

随着技术的不断发展进步&#xff0c;在前两年对我们还是天方夜谭&#xff0c;可望而不可即的科技&#xff0c;现在不仅在逐步实现&#xff0c;甚至在广泛地普及应用中。人工智能技术应用在越来越多的行业中&#xff0c;在一定程度上对各行业的发展起了不小的推动作用&#xff0…

教你如何将PDF文件转换成PPT演示文稿

在工作和学习中&#xff0c;我们可能需要将一些PDF文件转换成PPT演示文稿&#xff0c;以便于更好地展示和分享。虽然PPT和PDF是两种不同的文档格式&#xff0c;但是我们可以使用一些专业的软件或在线工具来实现这种转换。下面就让我们来教你如何将PDF文件转换成PPT演示文稿。 …

Java 简介

Java 是由 Sun Microsystems 公司于 1995 年 5 月推出的 Java 面向对象程序设计语言和 Java 平台的总称。由 James Gosling和同事们共同研发&#xff0c;并在 1995 年正式推出。 后来 Sun 公司被 Oracle &#xff08;甲骨文&#xff09;公司收购&#xff0c;Java 也随之成为 O…

RabbitMQ 发布确认 交换机 死信队列 延迟队列

RabbitMQ 发布确认开启发布确认的方法单个确认发布批量消息确认发布异步确认发布如何处理异步未确认消息 交换机绑定Fanout交换机Fannout交换机&#xff08;消费者&#xff09;Fannout交换机&#xff08;生产者&#xff09; Direct exchage(直接交换机)生产者消费者 Topic交换机…

遥感影像变化检测新方法:MLDANets

来源&#xff1a;投稿 作者&#xff1a;xin 编辑&#xff1a;学姐 论文标题&#xff1a; Multilevel Deformable Attention-Aggregated Networks for Change Detection in Bitemporal Remote Sensing Imagery Motivation 本文指出: &#xff08;1&#xff09;当前基于自注意…