C语言-扫雷游戏的实现

🌈write in front🌈
🧸大家好,我是Aileen🧸.希望你看完之后,能对你有所帮助,不足请指正!共同学习交流.
🆔本文由Aileen_0v0🧸 原创 CSDN首发🐒 如需转载还请通知⚠️
📝个人主页:Aileen_0v0🧸—CSDN博客
🎁欢迎各位→点赞👍 + 收藏⭐️ + 留言📝​
📣系列专栏:Aileen_0v0🧸的C语言学习系列专栏——CSDN博客
🗼我的格言:"没有罗马,那就自己创造罗马~"

目录

1.扫雷游戏的分析和设计👻

1.1扫雷游戏的功能说明💭

1.2游戏的界面▶️​编辑

2.扫雷游戏的代码实现🔆

2.1数据结构的分析🙉

​编辑

2.2文件结构设计💫

2.3游戏的过程实现,代码块💦

主函数,用户菜单页面代码⏸️:

棋盘打印*️⃣:

给棋盘加坐标:🔢

布置雷💣:

排雷💥:

3.扫雷游戏的完整代码✨


1.扫雷游戏的分析和设计👻

1.1扫雷游戏的功能说明💭

使用控制台实现经典的扫雷游戏

游戏可以通过菜单实现继续玩或退出游戏
• 扫雷的棋盘是9*9的格子
• 默认随机布置10个雷
 可以排查雷

1.2游戏的界面▶️

初始界面

排雷界面

排雷失败界面

2.扫雷游戏的代码实现🔆

2.1数据结构的分析🙉

但是如果我们判断边缘的格子位置是否含雷时,

由于周围边界没有东西,导致我们需要判断这个格子是否位于边缘位置,这就会让代码变的复杂~

于是,我们可以通过 在原来9 * 9 的方格的 上下 , 左右位置 放没有雷的空格子(如上面左右两个数组的橙色边界)

根据刚刚的分析,

我们在左边创建一个mine数组 布置好雷的信息,全部初始化成 字符"0" ~

雷 - "1"

非雷 - "0"


在右边创建一个show数组放置 排查处的雷的信息, 最初未排查时,都放 *

没有排查 - "*"

排查 - 数字字符


小细节:之所以都用 字符数组 是因为 只需要定义字符函数, 方便操作~

如果 左边是整形数组,右边是字符数组 就 需要调用两个不同的函数~


在game.c中打印棋盘的时候,我们只打印9*9的~

因为外边的绿色空格只是为了编写变得容易一点,不会越界~~

但是,按照这种方式打印,我们很难知道是第几行第几列,因为没有标识

于是我们,再利用 for 循环打印出 行和列的序号 

利用库函数rand 随机布置雷

要从1到n中随机取一个数的公式是:


rand()%n+1.


解释一下:
1、库函数rand()会返回一个大于0的随机整数;
2、rand()%n,对这个返回的随机整数除以n取余,结果是一个0到n-1的随机整数

3、rand()%n+1,将rand()%n的结果加上1,就可得到一个1到n的随机整数;

更通用一点的公式,产生m到n中(n>m)的一个随机数的公式是:
rand()%(n-m+1)+m。

2.2文件结构设计💫

首先,先创建这三个文件.

2.3游戏的过程实现,代码块💦

主函数,用户菜单页面代码⏸️:
#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>//扫雷页面都实现
void menu()
{printf("*********************************\n");printf("******		1. play	    *****\n");printf("******		0. exit	    *****\n");printf("*********************************\n");
}int main()
{int input = 0;do{menu();printf("请选择:<");scanf("%d", &input);switch (input){case 1:printf("扫雷\n");break;case 0:printf("退出游戏\n");break;default:printf("选择错误,请重新选择\n");break;}//case 1 和 default 都是非零//里层switch语句走完 就会走while语句} while (input);return 0;//如果while 后面为0,程序就会自动退出游戏//非0   1就打印扫雷    其他值则重新打印菜单让用户选择}

运行效果:

棋盘打印*️⃣:

头文件game.h 的代码:

#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
#pragma once#define ROW 9
#define COL 9#define ROWS ROW + 2
#define COLS COL + 2// 函数的声明//初始化棋盘
void InitBoard(char board[ROWS][COLS], int rows, int cols,char set);//打印棋盘的
void DisplayBoard(char board[ROWS][COLS],int rows, int cols);

源文件game.c 的代码:

#define _CRT_SECURE_NO_WARNINGS#include "game.h"void InitBoard(char board[ROWS][COLS], int rows, int cols,char set)
{int i = 0;for (i = 0; i < rows; i++){int j = 0;for (j = 0; j < cols; j++){board[i][j] = set;}}
}//打印棋盘,就是打印数组
void DisplayBoard(char board[ROWS][COLS], int row, int col)
{int i = 0;for (i = 1; i <= row; i++){int j = 0;for (j = 1; j <= col; j++){printf("%c ", board[i][j]);}printf("\n");}
}

源文件 test.c 的代码:

#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
#include "game.h"//扫雷页面都实现
void menu()
{printf("*********************************\n");printf("******		1. play	    *****\n");printf("******		0. exit	    *****\n");printf("*********************************\n");
}void game()
{//数组char mine[ROWS][COLS];//"0"char show[ROWS][COLS];//"*"InitBoard(mine, ROWS, COLS,'0');InitBoard(show, ROWS, COLS,'*');//棋盘打印DisplayBoard(mine, ROW, COL);DisplayBoard(show, ROW, COL);//布置雷//排查雷}int main()
{int input = 0;do{menu();printf("请选择:<");scanf_s("%d", &input);switch (input){case 1:printf("扫雷\n");game();//游戏代码模块化break;case 0:printf("退出游戏\n");break;default:printf("选择错误,请重新选择\n");break;}//case 1 和 default 都是非零//里层switch语句走完 就会走while语句} while (input);return 0;//如果while 后面为0,程序就会自动退出游戏//非0   1就打印扫雷    其他值则重新打印菜单让用户选择}

 打印结果:

给棋盘加坐标:🔢

在原来打印棋盘上加上坐标,进行定位,只需修改game.c部分的代码:

#define _CRT_SECURE_NO_WARNINGS#include "game.h"void InitBoard(char board[ROWS][COLS], int rows, int cols,char set)
{int i = 0;for (i = 0; i < rows; i++){int j = 0;for (j = 0; j < cols; j++){board[i][j] = set;}}
}//打印棋盘,就是打印数组
void DisplayBoard(char board[ROWS][COLS], int row, int col)
{int i = 0;printf("-----------扫雷游戏-----------\n");//打印棋盘序号for (i = 0; i <= row; i++){printf("%d ", i);}printf("\n");//打印9*9的棋盘for (i = 1; i <= row; i++){printf("%d ", i);int j = 0;for (j = 1; j <= col; j++){printf("%c ", board[i][j]);}printf("\n");}printf("-----------扫雷游戏-----------\n");
}

打印结果:

布置雷💣:

game.h:

#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
#include <stdlib.h>//标准库头文件
#include<time.h>//布置雷
void SetMine(char mine[ROWS][COLS],int row , int col);
//虽然布置雷是在9*9的格子里面布置--->row 和 col
//但是,数组传参还是11*11的格子,即ROWS和COLS

game.c: 


//布置雷
void SetMine(char mine[ROWS][COLS], int row, int col)
{int count = EASY_COUNT;while (count){int x = rand() % row + 1;int y = rand() % col + 1;if (mine[x][y] == '0')//如果该位置无雷才在这个位置放雷{mine[x][y] = '1';count--;}}}

test.c文件: 

	//布置雷SetMine(mine, ROW, COL);DisplayBoard(mine, ROW, COL);

 打印结果:

排雷💥:

通过观察,ASCII表可知:

字符'0'-->ASCII值:48

字符'1'-->ASCII值:49

字符'2'-->ASCII值:50

字符'3'-->ASCII值:51

依此类推

得出规律:'1' - '0' = 49-48 = 1

                '3' - '0' = 51-48 = 3

字符-字符=数字   ----------> 反推: 数字+字符=字符

统计 x y 周围有几个雷 --->

把其周围的字符值'0'和'1'加起来即可

然后减去8个字符'0' 得到数字,去代替  x y 处的 '*'.

game.h:

//排查雷
void FindMine(char mine[ROWS][COLS], char show[ROWS][COLS], int row, int col);

game.c:


//实现GetMineCount数组
static int GetMineCount(char mine[ROWS][COLS], int x, int y)
{return(mine[x - 1][y] +mine[x - 1][y - 1] +mine[x][y - 1] +mine[x + 1][y - 1] +mine[x + 1][y] +mine[x + 1][y + 1] +mine[x][y + 1] +mine[x - 1][y + 1] - 8 * '0');
}
//排查雷
void FindMine(char mine[ROWS][COLS], char show[ROWS][COLS], int row, int col)
{int x = 0;int y = 0;while (1)//死循环排雷{printf("请输入要排查的坐标:>");scanf("%d %d", &x, &y);//注意:x y 要在有效的排查范围(9*9)之内if (x >= 1 && x <= row && y >= 1 && y <= col){//开始排查是否是雷if (mine[x][y] == '1'){printf("很遗憾,你被炸死了\n");DisplayBoard(mine, ROW, COL);break;}else{int count = GetMineCount(mine, x, y);show[x][y] = count + '0';DisplayBoard(show, ROW, COL);//count + 字符'0;变成对应的数字字符放到show数组里}}else{printf("坐标非法,重新输入\n");}}}

test.c:

	//排查雷FindMine(mine, show, ROW, COL);

注意:GetMineCount 没有在其它文件中声明是因为,我们只希望它在game.c处悄悄使用它,所以前加static

运行结果:

上面的排雷,未限制排雷次数,即可无限循环下去,这样子的游戏设计显然不合理~

于是,我们可以根据,雷和非雷的数量关系进行排雷循环次数的限制.

game.h:

//布置80个雷
#define EASY_COUNT 80

game.c: 

//排查雷
void FindMine(char mine[ROWS][COLS], char show[ROWS][COLS], int row, int col)
{int x = 0;int y = 0;int win = 0;while (win<row*col-EASY_COUNT)//根据雷和非雷的数量关系限制循环次数{printf("请输入要排查的坐标:>");scanf("%d %d", &x, &y);//注意:x y 要在有效的排查范围(9*9)之内if (x >= 1 && x <= row && y >= 1 && y <= col){//开始排查是否是雷if (mine[x][y] == '1'){printf("很遗憾,你被炸死了\n");DisplayBoard(mine, ROW, COL);break;}else{int count = GetMineCount(mine, x, y);show[x][y] = count + '0';DisplayBoard(show, ROW, COL);//count + 字符'0;变成对应的数字字符放到show数组里win++;}}else{printf("坐标非法,重新输入\n");}}if (win == row * col - EASY_COUNT){printf("恭喜你,扫雷成功\n");DisplayBoard(mine, ROW, COL);}}

我们可以通过改变雷的个数,然后根据mine的数组打印的结果对照着进行排雷成功的结果输出,检查是否有误. 

运行结果:

3.扫雷游戏的完整代码✨

game.h:

#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
#include <stdlib.h>//标准库头文件
#include<time.h>
#pragma once#define ROW 9
#define COL 9#define ROWS ROW + 2
#define COLS COL + 2//布置10个雷
#define EASY_COUNT 10// 函数的声明//初始化棋盘
void InitBoard(char board[ROWS][COLS], int rows, int cols,char set);//打印棋盘的
void DisplayBoard(char board[ROWS][COLS],int rows, int cols);

game.c:

#define _CRT_SECURE_NO_WARNINGS#include "game.h"
#include <stdio.h>void InitBoard(char board[ROWS][COLS], int rows, int cols,char set)
{int i = 0;for (i = 0; i < rows; i++){int j = 0;for (j = 0; j < cols; j++){board[i][j] = set;}}
}//打印棋盘,就是打印数组
void DisplayBoard(char board[ROWS][COLS], int row, int col)
{int i = 0;printf("-----------扫雷游戏-----------\n");//打印棋盘序号for (i = 0; i <= row; i++){printf("%d ", i);}printf("\n");//打印9*9的棋盘for (i = 1; i <= row; i++){printf("%d ", i);int j = 0;for (j = 1; j <= col; j++){printf("%c ", board[i][j]);}printf("\n");}printf("-----------扫雷游戏-----------\n");
}//布置雷
void SetMine(char mine[ROWS][COLS], int row, int col)
{int count = EASY_COUNT;while (count){int x = rand() % row + 1;int y = rand() % col + 1;if (mine[x][y] == '0')//如果该位置无雷才在这个位置放雷{mine[x][y] = '1';count--;}}}//实现GetMineCount数组
static int GetMineCount(char mine[ROWS][COLS], int x, int y)
{return(mine[x - 1][y] +mine[x - 1][y - 1] +mine[x][y - 1] +mine[x + 1][y - 1] +mine[x + 1][y] +mine[x + 1][y + 1] +mine[x][y + 1] +mine[x - 1][y + 1] - 8 * '0');
}
//排查雷
void FindMine(char mine[ROWS][COLS], char show[ROWS][COLS], int row, int col)
{int x = 0;int y = 0;int win = 0;while (win<row*col-EASY_COUNT)//根据雷和非雷的数量关系限制循环次数{printf("请输入要排查的坐标:>");scanf("%d %d", &x, &y);//注意:x y 要在有效的排查范围(9*9)之内if (x >= 1 && x <= row && y >= 1 && y <= col){//开始排查是否是雷if (mine[x][y] == '1'){printf("很遗憾,你被炸死了\n");DisplayBoard(mine, ROW, COL);break;}else{int count = GetMineCount(mine, x, y);show[x][y] = count + '0';DisplayBoard(show, ROW, COL);//count + 字符'0;变成对应的数字字符放到show数组里win++;}}else{printf("坐标非法,重新输入\n");}}if (win == row * col - EASY_COUNT){printf("恭喜你,扫雷成功\n");DisplayBoard(mine, ROW, COL);}}

test.c:

#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
#include "game.h"//扫雷页面都实现
void menu()
{printf("*********************************\n");printf("******		1. play	    *****\n");printf("******		0. exit	    *****\n");printf("*********************************\n");
}void game()
{//数组char mine[ROWS][COLS];//"0"char show[ROWS][COLS];//"*"InitBoard(mine, ROWS, COLS,'0');InitBoard(show, ROWS, COLS,'*');//棋盘打印//DisplayBoard(mine, ROW, COL);  雷的位置注释掉不打印出来,保持神秘感DisplayBoard(show, ROW, COL);//布置雷SetMine(mine, ROW, COL);//DisplayBoard(mine, ROW, COL);//排查雷FindMine(mine, show, ROW, COL);}int main()
{int input = 0;srand((unsigned int)time(NULL));//强制转换成无符号整型do{menu();printf("请选择:<");scanf_s("%d", &input);switch (input){case 1:printf("扫雷\n");game();//游戏代码模块化break;case 0:printf("退出游戏\n");break;default:printf("选择错误,请重新选择\n");break;}//case 1 和 default 都是非零//里层switch语句走完 就会走while语句} while (input);return 0;//如果while 后面为0,程序就会自动退出游戏//非0   1就打印扫雷    其他值则重新打印菜单让用户选择}

 

🌻今天的扫雷游戏就分享到这里啦~🌻

🌻喜欢就一键三连支持一下吧♥~🌻

🌻附上今天的日落图☺️🌻

🌻谢谢家人们!🌻

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

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

相关文章

CMD脚本实战教程

要在 Windows 11 上编写一个自定义关机的 CMD 脚本文件&#xff0c;你可以创建一个扩展名为 .bat 或 .cmd 的文本文件&#xff0c;并在其中编写脚本。 一、常用语法 rem&#xff1a;注释 pause&#xff1a;暂停正在执行的批处理文件&#xff0c;并提示用户按键之后继续执行 r…

ps丢失d3dcompiler_47.dll怎么办,这四个方法都能解决

在当今的信息化社会&#xff0c;电脑已经成为我们生活和工作中不可或缺的一部分。然而&#xff0c;随着软件技术的不断发展&#xff0c;电脑在使用过程中也难免会遇到各种问题。其中&#xff0c;缺失d3dcompiler_47.dll文件是一个常见的问题。本文将为大家介绍如何修复电脑出现…

git:二、git的本地配置+工作区域和文件状态+git add/commit/log +git reset回退版本

git的使用方式 命令行&#xff08;最常用&#xff09;图形化界面IDE插件/拓展&#xff08;次常用&#xff09; git的本地/系统配置 之前的文章提到过git的全局配置。如下&#xff1a; git config --global user.name "ss" git config --global user.email "…

计算机毕业设计 基于SSM+Vue的物资存储系统(以消防物资为例)的设计与实现 Java实战项目 附源码+文档+视频讲解

博主介绍&#xff1a;✌从事软件开发10年之余&#xff0c;专注于Java技术领域、Python人工智能及数据挖掘、小程序项目开发和Android项目开发等。CSDN、掘金、华为云、InfoQ、阿里云等平台优质作者✌ &#x1f345;文末获取源码联系&#x1f345; &#x1f447;&#x1f3fb; 精…

please choose a certificate and try again.(-5)报错怎么解决

the server you want to connect to requests identification,please choose a certificate and try again.(-5)

Centos7部署单机版MongoDB

目录 Centos7部署单机版MongoDBMongoDB介绍数据模型索引分布式高可用性查询语言驱动和社区用途缺点 下载并解压安装包创建相关文件夹和文件编辑mongod.conf文件启动mongodb创建管理员用户终止MongoDB服务配置自启动服务关闭SELinux编辑自启动服务文件mongodb服务命令 Centos7部…

坚鹏:中国邮政储蓄银行金融科技前沿技术发展与应用场景第4期

中国邮政储蓄银行金融科技前沿技术发展与应用场景第4期培训圆满结束 中国邮政储蓄银行拥有优良的资产质量和显著的成长潜力&#xff0c;是中国领先的大型零售银行。2016年9月在香港联交所挂牌上市&#xff0c;2019年12月在上交所挂牌上市。中国邮政储蓄银行拥有近4万个营业网点…

yolov5使用最新MPDIOU损失函数,有效和准确的边界盒回归的损失,优于GIoU/EIoU/CIoU/EIoU(附代码可用)

文章目录 1. 论文1.1. 主要目的1.2. 设计思路2 代码3.总结1. 论文 MPDIoU: A Loss for Efficient and Accurate Bounding Box Regression (一个有效和准确的边界框损失回归函数) 论文地址 1.1. 主要目的 当预测框与边界框具有相同的纵横比,但宽度和高度值完全不同时,大多数…

【一】Spring Cloud 系列简介

Spring Cloud 系列简介 简介&#xff1a;从单体架构到分布式架构&#xff0c;再到微服务架构&#xff0c;一路经历走来spring框架也一直在与时俱进&#xff0c;回顾下来感觉做Java开发就是基于spring开发&#xff0c;spring也一路发展出了spring boot&#xff0c;在此基础上发…

七天学会C语言-第七天(结构体)

1.定义结构体 例 1&#xff1a;把一个学生的信息(包括学号、姓名、性别、住址等 4 项信息) 放在一个结构体变量中&#xff0c;然后输出这个学生的信息。 #include <stdio.h>struct Student {int student_id;char name[30];char gender;char address[60]; };int main() …

flex布局与float布局

float布局 俩栏 三栏 flex布局

力扣刷题-数组-螺旋矩阵

模拟过程&#xff0c;但却十分考察对代码的掌控能力。 重点&#xff1a;循环不变量原则&#xff01; 第一条原则&#xff1a; 模拟顺时针画矩阵的过程&#xff1a; 填充上行从左到右填充右列从上到下填充下行从右到左填充左列从下到上 由外向内一圈一圈这么画下去。 第二条原…

基于微信小程序的加油站服务管理系统设计与实现(源码+lw+部署文档+讲解等)

文章目录 前言运行环境说明用户微信端的主要功能有&#xff1a;管理员的主要功能有&#xff1a;具体实现截图详细视频演示为什么选择我自己的网站自己的小程序&#xff08;小蔡coding&#xff09;有保障的售后福利 代码参考论文参考源码获取 前言 &#x1f497;博主介绍&#x…

八大排序(四)--------直接插入排序

本专栏内容为&#xff1a;八大排序汇总 通过本专栏的深入学习&#xff0c;你可以了解并掌握八大排序以及相关的排序算法。 &#x1f493;博主csdn个人主页&#xff1a;小小unicorn ⏩专栏分类&#xff1a;八大排序汇总 &#x1f69a;代码仓库&#xff1a;小小unicorn的代码仓库…

1.(vue3.x+vite)封装组件

前端技术社区总目录(订阅之前请先查看该博客) 关联博客 2.(vue3.x+vite)组件注册并调用 1:创建组件目录package,并创建相关工程结构 2:编写组件内容(index.vue) 3:添加注册组件方法(index.js) 4:添加路由

平滑加权轮询算法java实现

实现代码 /*** 功能描述: 平滑加权轮询算法** author zhang pu* date 11:46 2023/9/22*/public static void smoothnessWeightPollLoadBalance() {Server serverA new Server("127.0.0.1", 5, 0);Server serverB new Server("127.0.0.2", 3, 0);Server s…

深度学习从入门到入土

1. 数据操作 N维数组样例 N维数组是机器学习和神经网络的主要数据结构 0-d 一个类别&#xff1a; 1.0 1-d 一个特征向量(一维矩阵)&#xff1a;[1.0, 2.7, 3.4] 2-d 一个样本-特征矩阵-(二维矩阵) 3-d RGB图片 &#xff08;宽x高x通道&#xff09;- 三维数组 4-d 一个RGB…

React 全栈体系(十三)

第七章 redux 五、redux 异步编程 1. 理解 redux 默认是不能进行异步处理的,某些时候应用中需要在 redux 中执行异步任务(ajax, 定时器) 2. 使用异步中间件 npm install --save redux-thunk 3. 代码 - 异步 action 版 3.1 store /* src/redux/store.js */ /*** 该文件专…

大型集团借力泛微搭建语言汇率时区统一、业务协同的国际化OA系统

国际化、全球化集团&#xff0c;业务遍布全世界&#xff0c;下属公司众多&#xff0c;集团对管理方式和企业文化塑造有着很高的要求。不少大型集团以数字化方式助力全球统一办公&#xff0c;深化企业统一管理。 面对大型集团全球化的管理诉求&#xff0c;数字化办公系统作为集…

Matlab图像处理-模式识别

模式识别 模式识别就是用计算的方法根据样本的特征将样本划分到一定的类别中去。模式识别就是通过计算机用数学技术方法来研究模式的自动处理和判读&#xff0c;把环境与客体统称为“模式”。模式识别以图像处理与计算机视觉、语音语言信息处理、脑网络组、类脑智能等为主要研…