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

一、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);}}}elsereturn;
}

四、完成扫雷游戏的菜单

 

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.mzph.cn/news/627630.shtml

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

相关文章

【Linux】第二十九站:再谈进程地址空间

文章目录 一、一些疑问二、程序没有加载前的地址&#xff08;程序)三、程序加载后的地址四、动态库的地址 一、一些疑问 什么是虚拟地址&#xff1f;什么是物理地址&#xff1f;CPU读到的指令里面用的地址&#xff0c;是什么地址&#xff1f;&#xff1f; 我们之前在使用动态…

记录汇川:H5U与Factory IO测试14

现实53工位的物料运输。 设置了自动连续存启动&#xff1a;就是一个一个运&#xff0c;按照顺序将空的货架填满。 设置了自动连续存停止&#xff1a;就是完成当前循环后退出。 设置了自动连续取启动&#xff1a;就是一个一个运&#xff0c;按照顺序将有货的货架清空。 设置…

如何在Docker中搭建MinIO容器并实现无公网ip远程访问本地服务

文章目录 前言1. Docker 部署MinIO2. 本地访问MinIO3. Linux安装Cpolar4. 配置MinIO公网地址5. 远程访问MinIO管理界面6. 固定MinIO公网地址 前言 MinIO是一个开源的对象存储服务器&#xff0c;可以在各种环境中运行&#xff0c;例如本地、Docker容器、Kubernetes集群等。它兼…

Android Studio安装超详细步骤(包括SDK安装不成功,模拟器无法创建等问题)

本文主要介绍CPU为AMD锐龙和英特尔两种类型在安装中出现的一些问题&#xff0c;两种解决的方案不同&#xff0c;所以首先查看属于哪种&#xff0c;然后找相对应的安装方法。 Android Studio的安装需要准备两个安装文件&#xff0c;一个是java JDK ,一个是Android Studio均可在官…

kubesphere DevOps部署SpringCloud项目

&#x1f34e;devops部署SpringCloud项目 &#x1f345;环境说明&#x1f345;部署流程&#x1f9c1;创建DevOps工程&#x1f9c1;填写流水线信息&#x1f9c1;创建流水线 &#x1f345;部署应用所需脚本&#x1f9c1;jenkinsfile&#x1f9c1;Dockerfile&#x1f9c1;deploy.y…

如何才能把图片无损放大?这几个无损放大的方法分享给你们

在互联网的浩瀚海洋里&#xff0c;我们常常迷失在繁星般的图片之中&#xff0c;寻找那一款能映照出我们内心的头像。无论是渴望在社交媒体上熠熠生辉&#xff0c;还是在游戏世界中塑造独一无二的形象&#xff0c;一个与我们心灵相通的头像总能带给我们难以言喻的喜悦与满足。然…

【动态规划】【滑动窗口】【C++算法】 629K 个逆序对数组

作者推荐 【矩阵快速幂】封装类及测试用例及样例 本文涉及知识点 动态规划 C算法&#xff1a;滑动窗口总结 LeetCode629: K 个逆序对数组 逆序对的定义如下&#xff1a;对于数组 nums 的第 i 个和第 j 个元素&#xff0c;如果满足 0 < i < j < nums.length 且 nu…

0、C++预备知识

文章目录 C与C语言为什么要学习CC为什么难学c值得学习的新特征C参考书籍开发工具 可移植性和标准程序创建的技巧编译和链接 C与C语言 C是一种计算机高级程序设计语言&#xff0c;C既可以进行C语言的过程化程序设计&#xff0c;又可以进行以抽象数据类型为特点的基于对象的程序…

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

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

linux高级篇基础理论十二( 自动化运维工具Ansible )

♥️作者&#xff1a;小刘在C站 ♥️个人主页&#xff1a; 小刘主页 ♥️不能因为人生的道路坎坷,就使自己的身躯变得弯曲;不能因为生活的历程漫长,就使求索的 脚步迟缓。 ♥️学习两年总结出的运维经验&#xff0c;以及思科模拟器全套网络实验教程。专栏&#xff1a;云计算技…

vue-quill-editor 图片放大缩小 及富文本解析 放大图片预览

接上回 巴阿这是上一篇博客的标题 《vue vue-quill-editor 富文本编辑器 &#xff08;图片问题&#xff09;拦截粘贴动作 将粘贴的图片上传服务器 一个页面渲染多个富文本编辑器&#xff08;使用场景循环遍历 个数不定&#xff09;》 上篇传送门 需求1.编辑框里图片可放大缩…

什么是DNS(域名系统)

域名系统&#xff08;DNS&#xff09;像是互联网的电话簿&#xff0c;将人们容易记住的网址转换成计算机能理解的IP地址。 没有DNS&#xff0c;我们就需要记住复杂的数字序列来访问网站 DNS的基本概念 定义&#xff1a;DNS是一个分布式数据库&#xff0c;它将域名&#xff08…

阿里云服务器ECS性能与优势,为什么国内第一云?

为什么选择阿里云&#xff1f;阿里云服务器有哪些优势&#xff1f;阿里云全球第三&#xff0c;国内第一云&#xff0c;阿里云服务器网aliyunfuwuqi.com分享云服务器ECS在丰富ECS实例架构、弹性灵活、稳定可靠、便捷易用、安全保障和成本优化多方面优势&#xff1a; 阿里云服务…

BSP-D2000平台调试CM9434串口芯片

1.硬件原理图 原理图显示两块9434的INT分别接到D2000的GPIO0_A3和GPIO0_A5. 2.配置 2.1 设备树 gpio <&pio 1 1 1 1 1 0>;| | | | | | | || | | | | | | ------表示有效电平| | | | | | | | | …

走迷宫之推箱子

前言&#xff1a; 在上一篇文章当中我介绍了一个走迷宫的写法&#xff0c;但是那个迷宫没什么可玩性和趣味性&#xff0c;所以我打算在迷宫的基础上加上一个推箱子&#xff0c;使之有更好的操作空间&#xff0c;从而增强了游戏的可玩性和趣味性。 1. 打印菜单 void menu() {…

探索 Python:发现有趣的库——第 1 章:数据可视化之旅

在一个充满活力的科技世界中&#xff0c;数据分析专家“算法仙”和编程爱好者“代码侠”相遇了&#xff0c;决定一起踏上数据可视化的探险之旅。他们将运用 Matplotlib 和 Seaborn 这两个强大的 Python 库&#xff0c;将枯燥的数据转化为生动的图形。 算法仙&#xff1a;你好&…

Kafka 的 Consumer Group 解读

作为一份笔记&#xff0c;本文再次梳理一下 Kafka 的 Consumer Group。我们知道&#xff0c;一个 Topic 往往会有多个 Partition&#xff0c;一条消息只会被写到一个 Kafka 的 Partition 中&#xff0c;那 Consumer 是怎么消费 Message 的呢&#xff1f; Consumer Group 又从中…

如何低成本打造高效跨国企业内网

随着企业数字化转型的加速&#xff0c;企业对网络连接、安全性和可靠性的需求也变得越来越高&#xff0c;最近小编收到了好几封私信&#xff0c;咨询跨境企业内部组网方案。 投稿的是某跨国公司的IT运维人员&#xff0c;总部在海外&#xff0c;国内分支机构每天都需要进行数据互…

[Android]实现一个权限申请类

[Android]实现一个权限申请类 导言 在引入了动态权限申请之后&#xff0c;Android的权限申请就变得尤为繁琐&#xff0c;若是按照原有的方法一板一眼地进行申请&#xff0c;样板代码未免太多。因此本篇文章就使用ActivityResult API&#xff0c;来实现一个简单的权限申请类来帮…

C练习——魔术师猜三位数

题目&#xff1a; 有一种室内互动游戏&#xff0c;魔术师要每位观众心里想一个三位数abc&#xff08;a、b、c分别是百位、十位和个位数字&#xff09;&#xff0c;然后魔术师让观众心中记下acb、bac、bca、cab、cba五个数以及这5个数的和值。只要观众说出这个和是多少&#xf…