头文件定义、函数声明

下面就是扫雷中使用到的所有函数,为了省事我把所有的代码都放在一个C文件中实现

宏定义中设置了游戏的界面布局,以及设置地雷的个数(这里默认的是10个地雷),界面是一个9*9的方格布局

如图:

代码如下:

#include<stdio.h>
#include<stdlib.h>
#include<time.h> //布局9*9方格
#define ROW 9
#define COL 9 //多出隐藏的两行两列用来补全边缘棋盘6个方格凑9个方便计算
#define ROWS ROW + 2
#define COLS COL + 2 #define MINE_COUNT 10//地雷个数 /*******************************************************************************/ //游戏实现
void play();
//初始化界面布局
void InitBoard(char board[ROWS][COLS], int rows, int cols, char set);
//打印布局
void DisplayBoard(char board[ROWS][COLS], int rows, int cols);
//布置地雷
void SetMine(char board[ROWS][COLS], int row, int col);
//判断成功排除后剩下的方格数是否等于地雷数
int IsWin(char show[ROWS][COLS], int row, int col);
//使用递归函数来实现周围没地雷时展开多个
void spread(char mine[ROWS][COLS], char show[ROWS][COLS], int x, int y);
//计算周围地雷个数
int mine_count(char mine[ROWS][COLS], int x, int y);
//扫雷
void FindMine(char mine[ROWS][COLS], char show[ROWS][COLS], int row, int col);
//递归实现连续排除周围地雷
void spread(char mine[ROWS][COLS], char show[ROWS][COLS], int x, int y); /*******************************************************************************/

函数定义

游戏规则设定

这里使用了函数递归的方式来实现周围8个方格如果都没有地雷的话就一次性展开使用空格来填补,以及通过字符的ASCII码值来计算周围八个方格中地雷的数量以数字形式显示周围地雷个数。

接下来就可以进行简单的排雷操作了,输入坐标进行排雷,直到排完所有雷则通过游戏,如果输入的方格坐标为之前布置的雷坐标那么恭喜你踩到雷了游戏结束,你可以选择继续玩或者直接退出。

如图:

代码如下:

int IsWin(char show[ROWS][COLS], int row, int col)
{
int num = 0;
//排除一个地雷时便进行累加
for (int i = 1; i <= row; i++)
{
for (int j = 1; j <= col; j++)
{
if (show[i][j] == '*')
num++;
}
}
return num;
} void spread(char mine[ROWS][COLS], char show[ROWS][COLS], int x, int y)
{
int count = mine_count(mine, x, y);//周围地雷的个数(此时是字符的ASCII码值) if (count == 0)
{
show[x][y] = ' ';
int i = 0, j = 0; for (i = -1; i <= 1; i++)
{
for (j = -1; j <= 1; j++)
{
//连续排除时限制范围在棋盘范围内
if ((x + i) > 0 && (y + i) > 0 && (x + i < ROWS) && (y + j < COLS) && show[x + i][y + j] == '*')
{
spread(mine, show, x + i, y + j);//递归实现周围如果都没地雷连续排除
}
}
}
}
else
{
show[x][y] = count + '0';//将字符ASCII码值转换为数字从而显示
}
} int mine_count(char mine[ROWS][COLS], int x, int y)
{
//将周围八个方格中字符的ASCII码值之和减去八个‘0’的值得到周围地雷个数
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, y = 0; while (1)
{
printf("请输入排雷坐标:>");
loop:
scanf("%d%d", &x, &y);
if (x >= 1 && x <= row && y >= 1 && y <= col)//判断坐标是否合法
{
if (mine[x][y] == '1')//‘1’为地雷
{
printf("你已踩到雷,游戏结束\n");
DisplayBoard(mine, ROW, COL);
break;
}
else
{
spread(mine, show, x, y);//计算周围地雷数量,连续排除无雷的方格
DisplayBoard(show, ROW, COL);
}
}
else
{
printf("请输入正确的坐标:");
goto loop;
}
int ret = IsWin(show, row, col);
if (ret == MINE_COUNT)//当累加的地雷数量等于布置地雷的数量则说明地雷全部排除
{
printf("恭喜你,通关成功!\n");
break;
}
}
} /*******************************************************************************/

布置地雷

这里使用了rand()函数在该方格没有地雷的情况下随机布置相应的地雷,有地雷的方格值为1,没有地雷的方格值为0,布置完地雷后再使用打印函数将界面布局打印出来

扩展:

在C语言中,我们一般使用 <stdlib.h> 头文件中的rand()函数来生成随机数

int rand (void);        //void 表示不需要传递参数。

但是多次运行上面的代码,你会发现每次产生的随机数都一样

实际上,rand() 函数产生的随机数是伪随机数,是根据一个数值按照某个公式推算出来的,这个数值我们称之为“种子”,我们可以通过 srand() 函数来重新“播种”,这样种子就会发生改变。

void srand (unsigned int seed);

它需要一个 unsigned int 类型的参数。在实际开发中,我们可以用时间作为参数,只要每次播种的时间不同,那么生成的种子就不同,最终的随机数也就不同。

使用<time.h>头文件中的 time() 函数即可得到当前的时间(精确到秒),就像下面这样:

srand((unsigned)time(NULL));

代码如下:

void SetMine(char board[ROWS][COLS], int row, int col)
{
int count = MINE_COUNT;//布置地雷数量
while (count)
{
//当该方格内没有地雷时随机布置相应数量地雷
int x = rand() % row + 1;//(行,随机数范围1~9)
int y = rand() % col + 1;//(列,随机数范围1~9)
if (board[x][y] == '0')//判断没有地雷
{
board[x][y] = '1';//布置地雷
count--;//直到布置完所有地雷
}
}
} void DisplayBoard(char board[ROWS][COLS], int row, int col)
{
int i = 0, j = 0; for (i = 0; i <= col; i++)//打印列号
{
printf(" %d ", i);
}
printf("\n");
for (i = 1; i <= row; i++)
{
printf(" %d ", i);//打印行号
for (j = 1; j <= col; j++)
{
printf(" %c ", board[i][j]);//打印9*9方格内字符布局
}
printf("\n");
}
printf("\n");
} void InitBoard(char board[ROWS][COLS], int rows, int cols, char set)
{
int i = 0, j = 0;
for (i = 0; i < rows; i++)
{
for (j = 0; j < cols; j++)
{
board[i][j] = set;//初始化所有方格内的字符
}
}
} /*******************************************************************************/

游戏实现函数

首先初始化游戏布局并打印出来,之后进行布置地雷以及计算周围八个方格中地雷的数量,最后开始扫雷

如图:

代码如下:

void play()
{
//随机数种子,time函数的返回值生成时间戳,保证每次随机布置的地雷位置不同
srand((unsigned int)time(NULL));
//地雷信息的存储
char mine[ROWS][COLS] = { 0 };//布置好的地雷信息11*11
char show[ROWS][COLS] = { 0 };//排查出的地雷信息11*11 InitBoard(mine, ROWS, COLS, '0');//将方格置为字符‘0’也就是无地雷状态
InitBoard(show, ROWS, COLS, '*');//将方格内字符初始化为*达到隐藏效果
DisplayBoard(show, ROW, COL); //DisplayBoard(mine,ROW,COL);//显示地雷位置
SetMine(mine, ROW, COL);
FindMine(mine, show, ROW, COL);
} /*******************************************************************************/

菜单函数以及主函数定义

这里是进入游戏的一个简单的选择菜单界面,通过选择(1/2)来执行相应的操作,调用该游戏实现函数以及退出游戏和错误判断

如图:

代码如下:

int main()
{
//菜单选择界面
int input = 0, choose = 0;
printf("&---------------------------&\n");
printf("| 1、扫雷 |\n");
printf("|---------------------------|\n");
printf("| 2、退出 |\n");
printf("&---------------------------&\n");
printf("请选择(1/2)>:");
do
{
scanf("%d", &input);
switch (input)
{
case 1:
while (1)
{
printf("\n");
play();
printf("&---------------------------&\n");
printf("| 1、继续 |\n");
printf("|---------------------------|\n");
printf("| 2、退出 |\n");
printf("&---------------------------&\n");
printf("请输入你的选择:");
loop:
scanf("%d", &choose);
switch (choose)
{
case 1:
break;
case 2:
return 0;
default:
printf("请输入正确的选项:");
goto loop;
}
}
case 2:
printf("退出成功!\n");
return 0;
default:
printf("请输入正确的选项:");
break;
}
} while (input); return 0;
} /*******************************************************************************/

总结

  以上就是个人分享的关于扫雷游戏所有的代码,用C语言简单的实现了扫雷这个小游戏,虽然在界面和功能的实现上都很粗糙,但是对于一个初学C语言小白的我来讲也是一个小小的进步,也希望自己继续加油之后写出更优秀的代码

C语言实现扫雷游戏(完整版)的更多相关文章

  1. 爱拼图游戏android源码完整版

    这个是一款爱拼图游戏源码完整版,该游戏源码比较完整的,可以支持音乐的播放在游戏的玩的过程中,还可以控制系统的声音等,可以支持多种图片的选择来进行玩的,还可以根据自己的爱好选择不同的难度来的,级别分为: ...

  2. 哆啦A梦连连看游戏源码完整版

    这个源码是哆啦A梦连连看游戏源码完整版,也是安卓教程网android.662p.com分享过来的,哆啦A梦大家一定再熟悉不过了,这次登场的角色你能认出全部吗?赶紧把相同的小图标全部消除吧,一起体验下! ...

  3. Android版的疯狂猜图游戏源码完整版分享

    这个游戏源码是在安装教程网那么分享过来的,Android版的疯狂猜图游戏源码完整版分享,也是本人之前很早以前发的一款游戏源码的,大家如果想了解一下,可以看看吧,不说多了,上一个图先吧.   > ...

  4. Android宝宝点点乐游戏源码完整版

    Android宝宝点点乐游戏源码完整版,是我从其他网站转载过来的,就是那个安卓教程网,该游戏目前已经上线了百度应用商店了,大家可以去搜索一下,看看演示吧. <ignore_js_op> & ...

  5. 数据结构系列之2-3-4树的插入、查找、删除和遍历完整版源代码实现与分析(dart语言实现)

    本文属于原创,转载请注明来源. 在上一篇博文中,详细介绍了2-3树的操作(具体地址:https://www.cnblogs.com/outerspace/p/10861488.html),那么对于更多 ...

  6. Visual Studio 2010 SP1 中文升级补丁ISO完整版下载 (含多国语言)

    最近事情比较多,差点忘记了这个VS2010 SP1的升级补丁更新了,程序员们赶快更新吧!这次发布的SP1包含了下列语言:英文,简体中文,繁体中文,法语,德语,印度语,日语,韩语,俄语以及西班牙语. V ...

  7. C语言二维数组实现扫雷游戏

    #include<stdio.h> //使用二维数组实现 扫雷 int main() { char ui[8][8]={ '+','+','+','+','+','+','+','+', ...

  8. Android掌中游斗地主游戏源码完整版

    源码大放送-掌中游斗地主(完整版),集合了单机斗地主.网络斗地主.癞子斗地主等,有史以来最有参考价值的源码,虽然运行慢了一点但是功能正常,用的是纯java写的. 项目详细说明:http://andro ...

  9. JAVA在线观看视频教程完整版

    今天给大家介绍一下JAVA在线观看视频教程完整版,我们知道Java是一种可以撰写跨平台应用软件的面向对象的程序设计语言,是由Sun Microsystems公司于1995年5月推出的Java程序设计语 ...

随机推荐

  1. Apache DolphinScheduler ASF 孵化器毕业一周年,汇报来了!

    不知不觉,Apache DolphinScheduler 已经从 Apache 软件基金会(以下简称 ASF)孵化器毕业一年啦! 北京时间 2021 年 4 月 9 日,ASF 官方宣布 Apache ...

  2. 如何实现 System.out.println("a") 显示 b

    今天看到一篇文章不用反射,能否交换两个字符串的值. 心想字符串常量在常量池里面,是在就算用了反射也交换不了吧.转念一想,不对,字符串常量虽然本身在常量池里面,但是它依然是个对象,那么 private ...

  3. Taurus.MVC 微服务框架 入门开发教程:项目集成:2、客户端:ASP.NET Core(C#)项目集成:应用中心。

    系列目录: 本系列分为项目集成.项目部署.架构演进三个方向,后续会根据情况调整文章目录. 本系列第一篇:Taurus.MVC V3.0.3 微服务开源框架发布:让.NET 架构在大并发的演进过程更简单 ...

  4. jbd2的死锁分析

    已经运行多年的jbd2,它还是死锁了 背景:这个是在centos7的环境上复现的,内核版本为3.10.0-957.27.2.el7 下面列一下我们是怎么排查并解这个问题的. 一.故障现象 oppo云内 ...

  5. Spring mvc源码分析系列--前言

    Spring mvc源码分析系列--前言 前言 距离上次写文章已经过去接近两个月了,Spring mvc系列其实一直都想写,但是却不知道如何下笔,原因有如下几点: 现在项目开发前后端分离的趋势不可阻挡 ...

  6. 【java】学习路线7-继承、super方法、重写、重载

    /*继承-java只有单继承如果你创建了很多个class,但是之间有很多相同的成员变量和成员方法,修改的时候又要多处修改好麻烦,此时就可以创建多一个类来存储这些重复的东西,统一管理.相当方便.*//* ...

  7. MyBatis快速上手与知识点总结

    目录 1.MyBatis概述 1.1 MyBatis概述 1.2 JDBC缺点 1.3 MyBatis优化 2.MyBatis快速入门 3.Mapper代理开发 3.1 Mapper代理开发概述 3. ...

  8. 经纬度转换为距离单位km的方法

    function rad(d){ return d * Math.PI /180.0; }; GetDistance(lat1, lng1, lat2, lng2){ var radLat1 =rad ...

  9. Python中None作为索引的作用

    None的作用主要是在使用None的位置新增一个维度. a = np.arange(25).reshape(5,5) print(a) ''' [[ 0 1 2 3 4] [ 5 6 7 8 9] [ ...

  10. dp-LIS LCS 模型

    最长上升子序列问题: https://www.cnblogs.com/sxq-study/p/12303589.html 一:两遍LIS问题 1:题目: 怪盗基德是一个充满传奇色彩的怪盗,专门以珠宝为 ...