C语言之三子棋游戏实现篇

news2025/1/15 16:34:32

目录

主函数test.c

菜单函数

选择实现

游戏函数 (函数调用)

打印棋盘数据

打印展示棋盘 

玩家下棋

电脑下棋

判断输赢

 循环

test.c总代码 

头文件&函数声明game.h

头文件的包含

游戏符号声明

游戏函数声明

game.h总代码 

游戏函数game.c 

打印棋盘数据InitBoard

打印展示棋盘DisplayBoard 

打印数据行

打印分割行

玩家下棋PlayerMove

展示棋盘DisplayBoard 

电脑下棋ComputerMove

展示棋盘DisplayBoard 

判断输赢IsWin

判断平局IsFull

 game.c总代码


拖了许久的三子棋游戏的实现,今天来一步一步的实现三子棋游戏。🙂

主函数test.c

菜单函数

void menu()
{
	printf("***************************\n");
	printf("*******   1.Play   ********\n");
	printf("*******   0.over   ********\n");
	printf("***************************\n");
}

选择实现

void menu()
{
	printf("***************************\n");
	printf("*******   1.Play   ********\n");
	printf("*******   0.over   ********\n");
	printf("***************************\n");
}
void game()
{
	printf("开始游戏三子棋\n");
}
int main()
{
	//打印菜单
	int input = 0;
	do
	{
		menu();
		printf("请玩家选择:\n");
		scanf("%d", &input);
		switch (input)//分支语句
		{
		case 1://玩游戏
		{
			game();
			break;
		}
		case 0://结束游戏
		{
			printf("游戏结束\n");
			break;
		}
		default:
		{
			printf("输入错误,请重新选择\n");
			break;
		}
		}
	} while (input);
}

以上游戏的菜单和游戏选择我们曾在玩数字游戏中写过,大家可以回顾复习一下。

游戏函数 (函数调用)

打印棋盘数据

  • 打印棋盘,我们需要创建二维数组,存放棋盘数据。
char board[ROW][COL];//创建一个二维数组行和列
  • 我们需要创建一个函数InitBoard,功能是初始化棋盘数据。
InitBoard(board,ROW,COL);//把数组 数组的行和列都传过去

打印展示棋盘 

  • 打印棋盘 
DisplayBoard(board,ROW,COL);

玩家下棋

PlayerMove(board,ROW,COL);

电脑下棋

ComputerMove(board, ROW, COL);
srand((unsigned int)time(NULL));//产生随机数
//随机数的产生我们在前面猜数字游戏已经讲解过了,忘了的的小朋友可以去看看。

判断输赢

  • 玩家赢——函数返回'*'
  • 电脑赢——函数返回'#'
  • 平局——函数返回'Q'
  • 以上三种都跳出了循环,只有游戏继续——返回'C'——继续循环 
Iswin(board, ROW, COL);
char ret=0;
ret = Iswin(board, ROW, COL);
if (ret != 'C')
{
	break;
}
	if (ret == '*')
	{
		printf("玩家赢\n");
	}
	if (ret == '#')
	{
		printf("电脑赢\n");
	}
	if (ret == 'Q')
	{
		printf("平局\n");
	}

 循环

  • 每次玩家下棋完展示棋盘和判断输赢,同理电脑下棋完也要展示棋盘和判断输赢。
void game()
{
	char board[ROW][COL];
	InitBoard(board, ROW, COL);
	DisplayBoard(board, ROW, COL);
	//下棋循环
	char ret = 0;
	while (1)
	{
		PlayerMove(board, ROW, COL);//玩家下棋
		DisplayBoard(board, ROW, COL);//展示棋盘
		//判断输赢
		ret = Iswin(board, ROW, COL);
		if (ret != 'C')
		{
			break;
		}
		ComputerMove(board, ROW, COL);//电脑下棋
		DisplayBoard(board, ROW, COL);//展示棋盘
		//判断输赢
		ret = Iswin(board, ROW, COL);
		if (ret != 'C')
		{
			break;
		}
	}
	if (ret == '*')
	{
		printf("玩家赢\n");
	}
	if (ret == '#')
	{
		printf("电脑赢\n");
	}
	if (ret == 'Q')
	{
		printf("平局\n");
	}
}

test.c总代码 

#define _CRT_SECURE_NO_WARNINGS 1
#include"game.h"
void menu()
{
	printf("***************************\n");
	printf("*******   1.Play   ********\n");
	printf("*******   0.over   ********\n");
	printf("***************************\n");
}
//判断输赢
//玩家赢——返回'*'
//电脑赢——返回'#'
//平局——'Q'
// 以上三种都跳出了循环
//游戏继续——'C'
//继续循环
void game()
{
	char board[ROW][COL];
	InitBoard(board, ROW, COL);
	DisplayBoard(board, ROW, COL);
	//下棋循环
	char ret = 0;
	while (1)
	{
		PlayerMove(board, ROW, COL);//玩家下棋
		DisplayBoard(board, ROW, COL);//展示棋盘
		//判断输赢
		ret = Iswin(board, ROW, COL);
		if (ret != 'C')
		{
			break;
		}
		ComputerMove(board, ROW, COL);//电脑下棋
		DisplayBoard(board, ROW, COL);//展示棋盘
		//判断输赢
		ret = Iswin(board, ROW, COL);
		if (ret != 'C')
		{
			break;
		}
	}
	if (ret == '*')
	{
		printf("玩家赢\n");
	}
	if (ret == '#')
	{
		printf("电脑赢\n");
	}
	if (ret == 'Q')
	{
		printf("平局\n");
	}
}
int main()
{
	//打印菜单
	int input = 0;
	srand((unsigned int)time(NULL));//产生随机数
	do
	{
		menu();
		printf("请玩家选择:\n");
		scanf("%d", &input);
		switch (input)//分支语句
		{
		case 1://玩游戏
		{
			printf("开始游戏\n");
			game();
			break;
		}
		case 0://结束游戏
		{
			printf("游戏结束\n");
			break;
		}
		default:
		{
			printf("输入错误,请重新选择\n");
			break;
		}
		}
	} while (input);
}

头文件&函数声明game.h

头文件的包含

在我们写代码的过程中,会调用库函数,需要包含头文件,和声明函数。

所以我们将所有函数声明和头文件放到我们.h 文件中。

当然,在其他.c文件需要使用时,我们只需要包含 我们创造的 头文件"game.h" 即可。 

//#include"game.h"
#include<stdio.h>
#include<time.h>
#include<stdlib.h>

游戏符号声明

#define ROW 3
#define COL 3

游戏函数声明

void InitBoard(char board[ROW][COL],int row,int col);//打印棋盘数据
void DisplayBoard(char board[ROW][COL], int row, int col);//打印棋盘
void PlayerMove(char board[ROW][COL], int row, int col);
void ComputerMove(char board[ROW][COL], int row, int col);
char Iswin(char board[ROW][COL], int row, int col);
int IsFull(char board[ROW][COL], int row, int col);

game.h总代码 

#define _CRT_SECURE_NO_WARNINGS 1
#pragma once
#include<stdio.h>
#include<time.h>
#include<stdlib.h>
#define ROW 3
#define COL 3
void InitBoard(char board[ROW][COL], int row, int col);
void DisplayBoard(char board[ROW][COL], int row, int col);
void PlayerMove(char board[ROW][COL], int row, int col);
void ComputerMove(char board[ROW][COL], int row, int col);
char Iswin(char board[ROW][COL], int row, int col);
int  IsFull(char board[ROW][COL], int row, int col);

游戏函数game.c 

分析一下棋盘。

  • 棋盘分为 数据行分割行
  • 数据行,落子在棋盘上,每一个棋盘上的位置对应二维数组中的一个数组元素。在还未 落子   在棋盘上的时候,最开始棋盘上的数据初始化为空格
  •  分割行,有行和列。

分析完棋盘,我们要将棋盘打印出来,也就是把数据行和分割行打印出来,我们需要拆分 。

  • 数据行,我们拆分为  空格
  • 分割行,我们拆分为   ----

注意:分割行没有最后一列和最后一行,用分支语句来控制。 

打印棋盘数据InitBoard

  • 我们将数组的参数传递给了函数InitBoard。
  • 接下来,利用遍历二维数组元素,把数组中的元素初始化(用赋值运算符)为 空格 
  • 初始化空格,注意是' '单引号
#include"game.h"
void InitBoard(char board[ROW][COL], int row, int col)
{
	int i = 0;
	for (i = 0; i < row; i++)
	{
		int j = 0;
		for (j = 0; j < col; j++)
		{
			board[i][j] = ' ';//初始化为空格//字符单引号!
		}
	}
}

打印展示棋盘DisplayBoard 

  • 注意要换行
  • 注意数据行和分割行的空位大小要一致
  • 注意使用行和列循环,棋盘被改变格式3*3改为10*10,也不会错乱。

打印数据行

void DisplayBoard(char board[ROW][COL], int row, int col)
{
	int i = 0;
	int j = 0;
	for (i = 0; i < row; i++)//控制行一行一行打印
	{
		//打印数据行
		int j = 0;//每行的元素按列打印
		for (j = 0; j < col; j++)
		{
			printf("   ", board[i][j]);//打印数据//初始化为空格//三个空格
			if (j < col - 1)
				printf("|");//打印分割线
		}

打印分割行

		//打印分割行
		printf("\n");
		if (i < row - 1)
		{
			int j = 0;
			for (j = 0; j < col; j++)
			{
				printf("---");//对应数据行的三个空格
				if (j < col - 1)
					printf("|");
			}
		}
	}
}

 棋盘效果如下:

玩家下棋PlayerMove

  •  注意设置限制条件_不能超出棋盘范围
  •  注意棋盘是数组,数组的下标是从0开始的。
  •  注意循环,下棋成功break跳出循环。
void PlayerMove(char board[ROW][COL], int row, int col)
{
	int x = 0;
	int y = 0;
	while (1)
	{
		//先让玩家下棋
		printf("请输入要下棋的坐标:");
		scanf("%d %d", &x, &y);
		//不能超出棋盘范围
		if (x >= 1 && x <= row && y <= col && y>=1)
		{
			if (board[x - 1][y - 1] == ' ')//数组下标要从开始
			{
				board[x - 1][y - 1] = '*';
			}
			else
			{
				printf("该坐标已被占用,请输入其他坐标\n");
			}
		}
		else
		{
			printf("坐标超出棋盘范围,请重新输入\n");
		}
	}
}

展示棋盘DisplayBoard 

电脑下棋ComputerMove

  • 随机数的产生要掌握🆗 
void ComputerMove(char board[ROW][COL], int row, int col)
{
	printf("电脑下棋\n");
	int x = 0;
	int y = 0;
	while (1)
	{
		x = rand() % row;
		y = rand() % col;//产生随机值
		if (board[x][y] == ' ')
		{
			board[x][y] = '#';
			break;
		}
	}
}

展示棋盘DisplayBoard 

判断输赢IsWin

  • 玩家赢——函数返回'*'
  • 电脑赢——函数返回'#'
  • 平局——函数返回'Q'
  • 以上三种都跳出了循环,只有游戏继续——返回'C'——继续循环 

产生以上的结果的条件,我们来分析以下:

char Iswin(char board[ROW][COL], int row, int 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][1] != ' ')//循环1
		{
			return board[i][1];
		}
		if (board[0][i] == board[1][i] && board[1][i] == board[2][i] && board[1][i] != ' ')//循环2
		{
			return board[1][i];
		}
		if (board[0][0] == board[1][1] && board[1][1] == board[2][2] && board[1][1] != ' ')//循环3
		{
			return board[1][1];
		}
		if (board[2][2] == board[1][1] && board[1][1] == board[0][2] && board[1][1] != ' ')
		{
			return board[1][1];
		}
		if (IsFull(board, row, col))//返回1真执行,返回0假不执行
		{
			return 'Q';
		}
		//如果以上情况均不是
		return 'C';
	}
}

判断平局IsFull

if (IsFull(board, row, col))//返回1真执行,返回0假不执行
{
	return 'Q';
}
int IsFull(char board[ROW][COL], int row, int col)
{
	int i = 0;
	int j = 0;
	for (i = 0; i < row; i++)
	{
		for (j = 0; j < col; j++)
		{
			if (board[i][j] == ' ')
				return 0;//有空位不执行
		}
	}
	return 1;
}

 game.c总代码

#define _CRT_SECURE_NO_WARNINGS 1
#include"game.h"
void InitBoard(char board[ROW][COL], int row, int 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 row, int col)
{
	int i = 0;
	int j = 0;
	for (i = 0; i < row; i++)//控制行一行一行打印
	{
		//打印数据行
		int j = 0;//每行的元素按列打印
		for (j = 0; j < col; j++)
		{
			printf(" %c ", board[i][j]);//打印数据//初始化为空格
			if (j < col - 1)
				printf("|");//打印分割线
		}
		//打印分割行
		printf("\n");
		if (i < row - 1)
		{
			int j = 0;
			for (j = 0; j < col; j++)
			{
				printf("---");
				if (j < col - 1)
					printf("|");
			}
		}
		printf("\n");
	}
}

void PlayerMove(char board[ROW][COL], int row, int col)
{
	int x = 0;
	int y = 0;
	while (1)
	{
		//先让玩家下棋
		printf("请输入要下棋的坐标:");
		scanf("%d %d", &x, &y);
		//不能超出棋盘范围
		if (x >= 1 && x <= row && y <= col && y>=1)
		{
			if (board[x - 1][y - 1] == ' ')//数组下标要从开始
			{
				board[x - 1][y - 1] = '*';
				break;
			}
			else
			{
				printf("该坐标已被占用,请输入其他坐标\n");
			}
		}
		else
		{
			printf("坐标超出棋盘范围,请重新输入\n");
		}
	}
}


void ComputerMove(char board[ROW][COL], int row, int col)
{
	printf("电脑下棋\n");
	int x = 0;
	int y = 0;
	while (1)
	{
		x = rand() % row;
		y = rand() % col;//产生随机值
		if (board[x][y] == ' ')
		{
			board[x][y] = '#';
			break;
		}
	}
}

char Iswin(char board[ROW][COL], int row, int 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][1] != ' ')//循环1
		{
			return board[i][1];
		}
		if (board[0][i] == board[1][i] && board[1][i] == board[2][i] && board[1][i] != ' ')//循环2
		{
			return board[1][i];
		}
		if (board[0][0] == board[1][1] && board[1][1] == board[2][2] && board[1][1] != ' ')//循环3
		{
			return board[1][1];
		}
		if (board[2][2] == board[1][1] && board[1][1] == board[0][2] && board[1][1] != ' ')
		{
			return board[1][1];
		}
		if (IsFull(board, row, col))//返回1真执行,返回0假不执行
		{
			return 'Q';
		}
		//如果以上情况均不是
		return 'C';
	}
}

int IsFull(char board[ROW][COL], int row, int col)
{
	int i = 0;
	int j = 0;
	for (i = 0; i < row; i++)
	{
		for (j = 0; j < col; j++)
		{
			if (board[i][j] == ' ')
				return 0;//有空位不执行
		}
	}
	return 1;
}

✔✔✔✔✔最后,感谢大家的阅读,如有不足和错误,欢迎指正。

代码---------→【gitee:https://gitee.com/TSQXG】

联系---------→【邮箱:2784139418@qq.cqq

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

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

相关文章

贝叶斯公式中的动词 命名技巧

一项血液化验有95%的把我诊断某种疾病&#xff0c;但是&#xff0c;这项化验用于健康人也会有1%的“伪阳性”结果(即如果一个健康人接受这项化验&#xff0c;则化验结果乌镇此人患有该疾病的概率是0.01)。如果该疾病的患者事实上只占总人口的0.5%&#xff0c;若某人化验结果为阳…

xfs ext4 结合lvm 扩容、缩容 —— 筑梦之路

ext4 文件系统扩容、缩容操作 扩容系统根分区 根文件系统在 /dev/VolGroup/lv_root 逻辑卷上&#xff0c;文件系统类型为ext4&#xff0c;大小为10G&#xff0c;现在要将其扩容成20G。 给空闲空间分区# 调整分区类型为LVM&#xff0c;也就是8e类型 fdisk /dev/sdb# 选定分区后使…

JVM 之字节码(.class)文件

本文中的内容参考B站尚硅谷宋红康JVM全套教程 你将获得&#xff1a; 1、掌握字节码文件的结构 2、掌握Java源代码如何在JVM中执行 3、掌握一些虚拟机指令 4、回答一些面试题 课程介绍 通过几个面试题初始字节码文件为什么学习class字节码文件什么是class字节码文件分析c…

【Spring MVC】

目录 &#x1f36e;1 什么是 MVC &#xff1f; &#x1f381;2 Spring MVC 的连接 &#x1f358;2.1 RequestMapping 实现 POST 和 GET 请求 &#x1f963;2.2 GetMapping 只支持 GET 请求 &#x1fad6;2.3 PostMapping 只支持 POST 请求 &#x1f36c;3 Spring MVC 获取参数的…

开始MySQL之路——外键关联和多表联合查询详细概述

多表查询和外键关联 实际开发中&#xff0c;一个项目通常需要很多张表才能完成。例如&#xff0c;一个商城项目就需要分类表&#xff0c;商品表&#xff0c;订单表等多张表。且这些表的数据之间存在一定的关系&#xff0c;接下来我们将在单表的基础上&#xff0c;一起学习多表…

第四方支付平台和聚合支付有什么区别?

第四方支付平台和聚合支付有什么区别&#xff1f; 聚合支付和第四方支付平台是移动支付领域的两种常见支付方式。它们在实际应用中有许多相似之处&#xff0c;给人们的生活带来了便利。然而&#xff0c;这两种支付方式也有本质的区别。我将从不同的角度对它们进行比较和分析。 …

找风景类视频素材就上这5个网站

免费高清的风景视频素材&#xff0c;我推荐你去这几个网站下载&#xff0c;赶紧收藏起来~ 菜鸟图库 https://www.sucai999.com/video.html?vNTYxMjky 菜鸟图库网素材非常丰富&#xff0c;网站主要还是以设计类素材为主&#xff0c;高清视频素材也很多&#xff0c;像风景、植…

回归预测 | MATLAB实现TSO-ELM金枪鱼群优化算法优化极限学习机多输入单输出回归预测(多指标,多图)

回归预测 | MATLAB实现TSO-ELM金枪鱼群优化算法优化极限学习机多输入单输出回归预测&#xff08;多指标&#xff0c;多图&#xff09; 目录 回归预测 | MATLAB实现TSO-ELM金枪鱼群优化算法优化极限学习机多输入单输出回归预测&#xff08;多指标&#xff0c;多图&#xff09;效…

java八股文面试[JVM]——垃圾回收器

jvm结构总结 常见的垃圾回收器有哪些&#xff1f; CMS&#xff08;Concurrent Mark Sweep&#xff09; 整堆收集器&#xff1a; G1 由于整个过程中耗时最长的并发标记和并发清除过程中&#xff0c;收集器线程都可以与用户线程一起工作&#xff0c;所以总体上来说&#xff0c;…

【Unity】拖拽放置模型时 为什么出现有时候有紧贴地面和有时候随机再空中的情况

&#x1f468;‍&#x1f4bb;个人主页&#xff1a;元宇宙-秩沅 hallo 欢迎 点赞&#x1f44d; 收藏⭐ 留言&#x1f4dd; 加关注✅! 本文由 秩沅 原创 &#x1f636;‍&#x1f32b;️收录于专栏&#xff1a;unity细节和bug &#x1f636;‍&#x1f32b;️优质专栏 ⭐【…

骨传导耳机对大脑有影响吗?骨传导耳机有什么副作用

先上结论&#xff0c;骨传导耳机对大脑没有影响。骨传导耳机使用的是骨传导技术&#xff0c;声音是通过头骨骨头和颌骨给内耳传递的&#xff0c;而不是通过传统的空气传播。 简单来说&#xff0c;骨传导技术使用人类骨骼结构和声学原理来传递声音&#xff0c;这种现象我们也很常…

深入剖析Kubernetes之Kubernetes的本质

文章目录 Kubernetes的本质 Kubernetes的本质 Kubernetes 项目在 Borg 体系的指导下&#xff0c;体现出了一种独有的“先进性”与“完备性”&#xff0c;而这些特质才是一个基础设施领域开源项目赖以生存的核心价值。 Kubernetes 项目的架构&#xff0c;跟它的原型项目 Borg 非…

Python中的API构建指南:在Flask中进行API开发

原文&#xff1a;Python中的API构建指南&#xff1a;在Flask中进行API开发 - 知乎 如何实现从一个软件与另一个软件的通信交互&#xff1f;就像我们的APP&#xff0c;如何实现微信支付、苹果支付&#xff1f; 其实&#xff0c;我们只需要一个API。 API&#xff08;应用程序编…

【二叉树入门指南】链式结构的实现

【二叉树入门指南】链式结构的实现 一、前置说明二、二叉树的遍历2.1前序遍历2.2中序遍历2.3 后序遍历 三、以前序遍历为例&#xff0c;递归图解四、层序遍历五、节点个数以及高度等5.1 二叉树节点个数5.2二叉树叶子节点个数5.3 二叉树第k层节点个数5.4 二叉树查找值为x的节点5…

黑客资料(基本概念,漏登平台,kali安装)

黑客资料 1.基本概念 1.1安全三要素 1.2 渗透测试 在拥有授权的前提下&#xff0c;模拟黑客的攻击手段进行测试&#xff0c;也被称为道德黑客 1.3 渗透测试的意义 1.4 渗透测试的流程 1.5 确定目标 这个确定目标&#xff0c;主要对范围进行测试&#xff0c;对那些设备&#…

【线程池】线程池拒绝策略还有这个大坑(二)

目录 踩坑代码 后果展示 原因 小结 概要 上文我们聊了聊阻塞队列&#xff0c;有需要的小伙伴可以去瞅瞅【线程池】换个姿势来看线程池中不一样的阻塞队列&#xff08;一&#xff09;_走了一些弯路的博客-CSDN博客 这波我们一起来研究下线程池的拒绝策略。 你肯定要说了&a…

odejs+vue+elementui个人图片电子相册网站_84ds3

系统阐述的是使用智能化电子相册系统的设计与实现&#xff0c;对于nodejs、B/S结构、MySQL进行了较为深入的学习与应用。主要针对系统的设计&#xff0c;描述&#xff0c;实现和分析与测试方面来表明开发的过程。开发中使用了vue.js框架和MySQL数据库技术搭建系统的整体架构。利…

gitcode中删除已有的项目

镜像地址&#xff1a; https://www.jianshu.com/p/504c1418adb7?v1693021320653 扩展阅读 如何在GitLab中删除一个项目 https://www.codenong.com/cs106866762/ 简介&#xff1a; 如何在GitLab中删除一个项目 最近GIT上建了太多项目。想清一下&#xff0c;就在网上查了查…

27台光刻机,ASML做出了选择,无法割舍中国市场

日前ASML公布的业绩显示二季度它对中国市场的光刻机出货量环比猛增两倍多至27台&#xff0c;显示出ASML对中国市场的重视&#xff0c;为何在荷兰决定自9月1日起限制对中国供应先进芯片设备之时&#xff0c;ASML却突然加大了对中国的光刻机供货力度呢&#xff1f; ASML虽然垄断着…

Ruoyi微服务启动流程

1、执行sql 执行sql ry-quarty.sql ry_2023706.sql 到ry-cloud 数据库 2、下载nacos 修改配置文件 修改连接地址 启动nacos 看到下面的配置文件即为成功 修改配置文件里面的数据库连接信息 3、修改nacos 为单机启动 4、启动项目即可 nacos自取 链接: https://pan.baidu…