扫雷游戏【可展开一片,超详细,保姆级别,此一篇足够】

news2025/1/11 5:58:50

一、C语言代码实现的扫雷游戏的运行

C语言实现扫雷

二、扫雷游戏的分析与设计

1.扫雷游戏的界面设计

在玩家玩扫雷的时候,它会给你一个二维的棋盘(下面的讲解都以9x9规格为例子),然后点击你想排查的坐标,若不是雷的,则显示周围雷的个数。

针对这个现象,我们很容易的会想到利用二维数组,根据我们想要的格式去打印一个棋盘出来。

2.扫雷游戏的布雷

布置的雷,要满足以下的的特征:产生的雷的坐标是随机的,产生的雷的坐标不能重复,产生的雷不能超过我们布置的棋盘规格。

针对这个问题,我们只要知道如何产生真正的随机数,以及分支与循环语句的的熟练,是很容易解决的。

3.扫雷游戏的结构分析 

我们如果把雷,和排查过的坐标信息,都保存在一个棋盘中,是不是不容易区分,又或者会产生歧义?

所以我们的设计是:用两个二维数组实现两个9x9的棋盘,一个棋盘用来布置雷,或非雷;另一个棋盘用来保存返回排查的坐标周围雷的个数。 为了下面的讲解通顺,我们将布置雷的棋盘记作:BoardMine;用来保存返回排查的坐标周围雷的个数的棋盘记作:BoardShow。

 4.扫雷游戏的规则制定

i.在扫雷游戏开始的时候,就已经对存储雷信息的棋盘进行好初始化,以及雷的布置;

ii.玩家输入要排查的坐标(x,y),判断x,y是否合法,不合法提示玩家,并让其重新输入

iii.判断BoardMine(x,y)是否有雷,若有雷,返回false,本次游戏结束。若无雷,计算周围的坐标雷的个数,并赋给BoardShow(x,y)

iiii.展开一片的规则,模块三详细介绍。

5.扫雷游戏的棋盘规模规划

在介绍扫雷游戏的规则制定的时候,提到过:计算周围的坐标雷的个数,可是若我们想要9x9的棋盘规格,就只定义9x9的二维数组。若对边界坐标的计算,就棘手了很多,那该怎么办呢?

6.文件结构的设计

Swpthund.h 用于放置库函数调用所需的头文件、预处理指令、扫雷游戏各种接口的声明

Swpthund.c 完成扫雷游戏各种接口的定义

test.c     完成扫雷游戏的菜单功能,玩家可通过菜单开始游戏,继续游戏,退出游戏。
7.结言

有了这些思维准备,那下面的功能函数代码的展示与讲解部分,则会一目了然。

三、扫雷游戏的功能函数的代码展示与讲解

1.棋盘的创建
在头文件中的定义,方便后续对棋盘规模的统一管理
//定义 行:ROW 列:COL ...
#define ROW 9
#define COL 9
#define ROWS COL+2
#define COLS ROW+2
#define THUNDER 10  //定义的雷的个数

2.打印棋盘+棋区的规划 
void BoardPrint(char(*zmh)[COLS], int x, int y)
{
	printf("----------------扫雷 游戏----------------\n\n");
	int i = 0;
	//打印横坐标的上界;
	for (i = 0; i <= y; i++)
	{
		printf("|---");
	}
	printf("|\n");
	//打印1~9的坐标;
	for (i = 0; i <= y; i++)
	{
		printf("|%-3d", i);
	}
	printf("|\n");
	//打印横坐标的下界;
	for (i = 0; i <= y; i++)
	{
		printf("|---");
	}
	printf("|\n");
	//打印雷区;
	for (i = 1; i <= x; i++)
	{
		int j = 0;	
		//打印纵坐标;
		printf("|%-3d", i);
		//打印每一行的雷区;
		for (j = 1; j <= y; j++)
		{
			printf("| %c ", zmh[i][j]);
		}
		printf("|\n");
		//打印雷区的下界;
		for (j = 0; j <= y; j++)
		{
			printf("|---");
		}
		printf("|\n");
	}
	printf("\n");
}

打印的效果如下:

是不是很好看,很有成就感?我也这样想的。 

3.棋盘的初始化

通过memset函数,将二维数组的内存,按照char类型,全部设置为Element字符。参数sz是二维数组的内存大小;x,y是二维数组的行 列 ;Element是二维数组要初始化的字符。

我的设计是,将雷区全部初始化为字符 '0' ,将玩家操作的棋盘全部初始化为字符 '*'。

为什么要这样设计呢?

1.如果我们雷区全部初始化为字符 ‘0’,而布置雷的时候,把雷的信息用 ‘1’ 表示。这样我们在后续计算的时候,可以利用ASCII码值,很容易求到雷的个数;

比如:

        '1' - '0' = 1        '2'-'0' = 2        2+'0' = '2'

2.把玩家操作的棋盘全部初始化为字符‘*’,表示为排查的坐标。

4.随机产生雷,并布置在BoardMine中

 

要传ROW,与COL的原因很简单,在开始,我们就说了我们只是为了方便处理棋盘边界坐标返回雷区的个数而行 列都+2,实际上真正要操作的还是未加之前的9x9的棋盘规模。

5.扫雷游戏的过程,以及判断游戏是否结束
_Bool BoardProcess(char(*zmh)[COLS], char(*lyy)[COLS], int x, int y)
{
	int m, n;//待玩家输入排查雷区的坐标
	int count = 0;//存储已排查坐标的个数
	while (count + THUNDER != x * y)
	{
		BoardPrint(lyy, ROW, COL);
		printf("请输入你要排查的位置:>");
		scanf("%d%d", &m, &n);
		//判断玩家输入坐标的合法性,以及是否重复输入:
		if (BoardJudge(lyy,x,y,m,n) == false)
		{
			printf("输入坐标非法,请重新输入\n");
		}
		else
		{
			if (zmh[m][n] == '1')
				return false;
			else
			{
				//因为如果靠函数返回值的话,这是个递归函数,逻辑上有点麻烦,所以沃传一个参数
				//用来接受已经排查到的坐标
				BoardUnfold(zmh, lyy, x, y, m, n, &count);
				printf("已经排查过的坐标:%d\n", count);
			}
		}
	}
	return true;
}

 这是判断玩家输入的坐标合法性的函数。

 通过Boardprocess的返回值来判断,游戏是输了,还是赢了。(这里用到了布尔类型,true表真,false表假)

6.返回排查坐标周围雷的个数

7.展开一片,并完成已排查坐标个数的统计
void BoardUnfold(char(*zmh)[COLS], char(*lyy)[COLS], int x, int y, int m, int n, int* pcin)
{
	//统计(m,n)周围雷个数,
	char count = BoardCheck(zmh,m,n);
	if (count != '0')
	{
		if (lyy[m][n] == '*')
		{
			lyy[m][n] = count;
			*pcin += 1;
		}
	}
	else if (lyy[m][n] != ' ')
	{
		lyy[m][n] = ' '; 
		*pcin += 1;
		for (int i = m - 1; i <= m + 1; i++)
		{
			for (int j = n - 1; j <= n + 1; j++)
			{
				if ((1 <= i && i <= x) && (1 <= j && j <= y))
					BoardUnfold(zmh, lyy, x, y, i, j, pcin);
			}
		}
	}
	else
		return;
}

四、完成扫雷游戏的菜单

 

void game()
{
	//创建雷区,玩家操作的棋区
	char BoardShow[ROWS][COLS] = { 0 };
	char BoardMine[ROWS][COLS] = { 0 };
	//初始化
	BoardInit(BoardMine, sizeof(BoardMine), ROWS, COLS, '0');
	BoardInit(BoardShow, sizeof(BoardShow), ROWS, COLS, '*');
	//布置雷
	BoardRand(BoardMine, ROW, COL);

	//打印棋盘:用于核验扫雷功能是否正常;
	//BoardPrint(BoardMine, ROW, COL);
	//BoardPrint(BoardShow, ROW, COL);
	
	//游戏过程;
	int ret = BoardProcess(BoardMine, BoardShow, ROW, COL);
	if (ret)
	{
		printf("恭喜你,扫雷游戏通过\n");
	}	
	else
	{
		printf("很遗憾,你被雷炸死了\n");
		BoardPrint(BoardMine, ROW, COL);
	}
}
int main()
{
	int input = 0;
	//设立rand的种子;一次程序,只需要调用一次srand就行
	srand((unsigned int)time(NULL));
	do {
		menu();
		printf("请输入下一步的指令>:");
		scanf("%d", &input);
		switch (input)
		{
		case 1:
			system("color 0B");
			game();
			break;
		case 0:
			system("cls");
			printf("你已经退出游戏\n");
			break;
		default:
			printf("输入错误,请重新输入\n");
		}
	} while (input);
	return 0;
}

五、扫雷游戏的源码

扫雷游戏/扫雷游戏 · 残风也想永存/C语言项目 - 码云 - 开源中国 (gitee.com)

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

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

相关文章

【23种设计模式应用场景汇总】

23种设计模式应用场景汇总 设计模式是一种在软件开发中解决特定问题的通用解决方案。下面我将尝试将23种设计模式融入到一个场景中&#xff1a; 假设我们正在开发一个在线购物系统&#xff0c;我们可以使用以下设计模式&#xff1a; 1. 工厂方法模式&#xff1a;当用户在网站上…

红米手机录屏功能在哪?这里有你想要的答案

“有人知道红米手机录屏功能在哪吗&#xff1f;刚买了最新款的红米K70&#xff0c;本来打算用来录制游戏&#xff0c;可是找了半天&#xff0c;就是没看见录屏功能&#xff0c;真的很着急&#xff0c;有没有大佬教教我。” 在手机成为人们生活不可或缺的一部分的今天&#xff…

市场下行,中国半导体进口数量、金额双双两位数锐减 | 百能云芯

根据中国海关总署最新统计&#xff0c;2023年中国累计进口集成电路&#xff08;半导体晶圆&#xff09;数量为4795亿颗&#xff0c;较2022年下降10.8%&#xff1b;而进口金额为3494亿美元&#xff0c;下降15.4%。这一数据显示&#xff0c;中国半导体进口在数量和金额两方面均出…

基于域账户及西门子simatic logon的集中权限管理的实现

原创 Luis Wang 智能大大号引文&#xff1a;博途工控人平时在哪里技术交流博途工控人社群 西门子在工业自动化领域可谓傲视群雄&#xff0c;在制药工厂&#xff0c;有超过50%以上的自动化系统、设备都配备了西门子的PLC、DCS、HMI及SCADA(wincc)平台&#xff1b;包括配液及CIP…

Arya碎碎念 | 2023年编程之旅,2024年代码人生展望

前言 回顾过去这一年&#xff0c; 2023年编程之旅&#xff0c;2024年代码人生展望 目录 前言回望2023写博客的缘起写博客的成就参加的创作活动产出的优质博客文章 展望2024代码开发方面博客规划方面 另外另外我有了一直喵 总结 回望2023 写博客的缘起 3年前&#xff0c;注册…

JMeter 源码解读HashTree

背景&#xff1a; 在 JMeter 中&#xff0c;HashTree 是一种用于组织和管理测试计划元素的数据结构。它是一个基于 LinkedHashMap 的特殊实现&#xff0c;提供了一种层次结构的方式来存储和表示测试计划的各个组件。 HashTree 的特点如下&#xff1a; 层次结构&#xff1a;Ha…

一键完成爬虫之Cookie获取:利用浏览器模拟一个cookie出来、面对反爬虫、加密的cookie的应对方法

一键完成爬虫之Cookie获取&#xff1a;利用浏览器模拟一个cookie出来、面对反爬虫、加密的cookie的应对方法 本文提供一个快速取得cookie的办法&#xff0c;用来应对一些网站的的反爬虫和cookie失效等情况本接口是收费的&#xff08;1分钱1次调用&#xff0c;不愿付费自行折腾…

flutter使用get依赖实现全局loading效果,弹窗loading状态

get dialog的官网文档&#xff1a;GetDialogRoute class - dialog_route library - Dart API 可以使用Get.dialog()方法来创建一个自定义的加载弹窗&#xff0c;get框架是支持自定义弹窗效果的&#xff0c;所以我们就使用这个方式来自定义一个弹窗效果&#xff0c;并且点击遮罩…

curl 测试返回状态码

curl 测试返回状态码 curl -I -m 10 -o /dev/null -s -w %{http_code} www.baidu.com -I 仅测试HTTP头 -m 10 最多查询10s -o /dev/null 屏蔽原有输出信息 -s silent -w %{http_code} 控制额外输出 测试&#xff1a; curl -I -m 10 -o /dev/null -s -w %{http_code} www.baid…

【上分日记】第379场周赛(分类讨论 + 数学 + 前缀和)

文章目录 前言正文1.3000. 对角线最长的矩形的面积2.3001. 捕获黑皇后需要的最少移动次数3.3002. 移除后集合的最多元素数3.3003. 执行操作后的最大分割数量 总结尾序 前言 终于考完试了&#xff0c;考了四天&#xff0c;也耽搁了四天&#xff0c;这就赶紧来补这场周赛的题了&a…

解决kali beef启动失败解问题

只限于出现这个提示的时候使用 卸载 ruby apt remove ruby 卸载 beef apt remove beef-xss 重新安装ruby apt-get install ruby apt-get install ruby-dev libpcap-dev gem install eventmachine 重新安装beef apt-get install beef-xss 弄完以上步骤如果还是不行就重启kali再试…

SpringBoot集成p6spy

P6Spy 是一个可以用来在应用程序中拦截和修改数据操作语句的开源框架。 通过 P6Spy 我们可以对 SQL 语句进行拦截,相当于一个 SQL 语句的记录器,这样我们可以用它来作相关的分析,比如性能分析。这里主要用于在控制台打印SQL时能自动将问号替换成实际参数打印一个可执行的SQL…

原子累加器 LongAdder

&#x1f47d;System.out.println(“&#x1f44b;&#x1f3fc;嗨&#xff0c;大家好&#xff0c;我是代码不会敲的小符&#xff0c;双非大四&#xff0c;Java实习中…”); &#x1f4da;System.out.println(“&#x1f388;如果文章中有错误的地方&#xff0c;恳请大家指正&a…

nginx基本优化

安装nginx隐藏版本号 查看百度web服务器 [rootcjq11 ~]# curl -I http://www.baidu.com 隐藏nginx服务器版本号 [rootcjq11 ~]# cd /usr/local/src/nginx-1.22.0/ [rootcjq11 nginx-1.22.0]# vim src/core/nginx.h第13、14行修改版本号和服务器名称 [rootcjq11 nginx-1.2…

Springboot开发的大学生寝室考勤系统刷脸进出宿舍系统源码有论文

主要功能&#xff1a; 学生可以申请换寝&#xff0c;申请宿舍维修&#xff0c;健康上报&#xff0c;寝室长可上报寝室考勤和补签考勤&#xff08; 正常签到&#xff0c;不在寝室&#xff0c;晚归&#xff09;&#xff0c;查看寝室通报&#xff0c;公告等。 宿管处需要学生刷脸进…

Android Studio 项目结构

manifests&#xff1a;用于存放安卓程序的配置文件 AndroidManifest.xml&#xff1a;这是Android应用程序的清单文件&#xff0c;包含了应用程序的基本信息和组件声明等java&#xff1a;Java源代码文件存放的根目录 主代码 com.example.app&#xff1a;应用程序的主包名&#x…

Matlab:isomorphism

语法&#xff1a; P isomorphism(G1,G2) %计算图G1和G2之间的图同构等价关系&#xff08;如果存在&#xff09;。若不存在同构&#xff0c;则P为空数组 P isomorphism(___,Name,Value) %使用一个或多个名称-值对组参数指定其他选项 [P,edgeperm] isomorph…

浅谈SQL优化

避免使用子查询 例&#xff1a; select * from t1 where id in (select id from t2 where name lolly1023);其子查询在MySQL5.5版本里&#xff0c;内部执行计划是&#xff1a;先查询外表再匹配内表&#xff0c;而不是先查内表t2&#xff0c;当外表的数据很大时&#xff0c;查…

VM安装群晖系统 挂载整个硬盘给群晖系统

前言 在我们日常业务需求中&#xff0c;经常需要把整个磁盘的空间分配给群晖使用&#xff0c;那么如何通过vm分配整个磁盘空间给群晖系统。 操作 需要用管理员运行VM虚拟机 然后添加硬盘 就可以有权限全部添加了。这样会清除要挂载的磁盘的全部的数据。

蓝莓产量预测(R语言版)

数据描述 字段名 描述 字段名 描述 id 蓝莓唯一标识 MinOfUpperTRange 花期内最高温带日平均气温的最低记录, Clonesize 蓝莓克隆平均大小 AverageOfUpperTRange 花期内最高温带日平均气温, Honeybee 蜜蜂密度 MaxOfLowerTRange 花期内最低温带日平均气温的最…