在练习过三子棋后,我们可以延伸一下制作一个简易的扫雷游戏。
同样的,我们先构建思路
1.扫雷是个什么样的游戏?
通过对表面的操作对,下层的雷进行排查 ,在对应坐标如果下层是雷则结束游戏,如果不是则展开周围一片的位置,对于这个程序的思路主要是要将其分层,分为可是层与存放雷的一层,
那么我们开始吧!
第一步首先是菜单显示
这里我们只需要加入一个do-while()循环就可以了
#include"game.h"
void menu()
{
printf("******************************\n");
printf("********* 1.play *********\n");
printf("********* 0.exit *********\n");
printf("******************************\n");
}
int main()
{
//设置随机数的种子
srand((unsigned int)time(NULL));//这个是后期我们进行随机雷的设置要用的
int input = 0;
do
{
menu();//菜单
printf("请选择:");
scanf("%d", &input);
switch (input)
{
case 1:
game();
break;
case 0:
printf("退出游戏!\n");
break;
default:
printf("输入错误,请重新输入!\n");
break;
}
} while (input);
return 0;
}
第二步进行游戏体的构建
创建一个数组mine为存放雷的,创建一个数组show为显示的,
创建一个函数用于初始化BoardInit(),在创建这个函数时我们设置一个接收字符set用于接收分别针对两个数组的字符,我们将雷盘初始值设置为‘0’,将显示的数组设置为‘*’,
接下来我们创建一个初始化雷的函数SetMine()在每一次进行游戏的时候初始化一批雷
最后就是非常关键的排查雷,这个在后面重点分析
void game()
{
//定义用于存放雷和显示雷的数组
char mine[ROWS][COLS];
char show[ROWS][COLS];
//数组初始化
BoardInit(mine, ROWS, COLS, '0');
BoardInit(show, ROWS, COLS, '*');
//埋雷
SetMine(mine, ROW, COL);
//打印雷盘
BoardPrint(show, ROW, COL);
//排雷
FindMine(mine, show, ROW, COL);
}
那么初步轮廓搭建好后之后我们就需要对各个函数及参数进行声明
#pragma once
#include<stdio.h>
#include<windows.h>
#include<time.h>
#include<stdlib.h>
#define ROW 9
#define COL 9
#define ROWS ROW + 2
#define COLS COL + 2
#define MINE 10
//数组初始化
void BoardInit(char board[ROWS][COLS], int rows, int cols, char set);
//埋雷
void SetMine(char board[ROWS][COLS], int row, int col);
//排雷
void FindMine(char mine[ROWS][COLS], char show[ROWS][COLS], int row, int col);
//打印雷盘
void BoardPrint(char board[ROWS][COLS], int row, int col);
那么我们开始将函数进行编写吧
1.数组初始化函数
这里需要注意的是,我们如果进行扫雷的范围为9*9的范围,那么我们在进行边界的判断时就会越界,如果将雷盘外一圈都初始化却不放雷,这样就不会产生越界的情况
//数组初始化
void BoardInit(char board[ROWS][COLS], int rows, int cols, char set)
{
int i = 0;
int j = 0;
for (i = 0; i < rows; i++)
{
for (j = 0; j < cols; j++)
{
board[i][j] = set; //set为初识化的字符这里我们初始化的时11*11的内容
}
}
}
2.埋雷(向mine中随机赋雷)
这里需要注意的是直接使用rand函数生产的是有规律的随机数,那么就需要我们在test的测试函数中使用srand(time(NULL));利用时间函数生产真正的随机数
//埋雷
void SetMine(char board[ROWS][COLS], int row, int col)
{
int count = MINE;
while (count)
{
int x = rand() % row + 1; //随机生成雷的坐标
int y = rand() % col + 1; //因为随机数与row(col)取余得到的值是0~8,但是我们需要的
是1~9
if (board[x][y] == '0') //检查该位置是否已经有雷
{
board[x][y] = '1';
count--;
}
}
}
3.打印雷盘
//打印雷盘
void BoardPrint(char board[ROWS][COLS], int row, int col)
{
int i = 0;
int j = 0;
printf("------扫雷游戏------\n");
for (i = 0; i <= row; 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]);
}
printf("\n");
}
printf("------扫雷游戏------\n");
}
以上解决完之后我们就需要处理最关键的与用户交互的过程——排雷
这里我们需要有一个记录非雷数量的变量来控制结束,pw指针将用于记录后面递归展开时,每展开一个非雷位置就加1,直到我们的所有非雷位置判断完成即游戏胜利,而后我们首先判断用户是否输入的值是越界的,如果不是进行下一步,再判定有没有重复进行选择,如果是新的坐标,那么我们再判断是不是雷,如果不是雷,那么我们就进行下一步,扩散展开
//排雷
void FindMine(char mine[ROWS][COLS], char show[ROWS][COLS], int row, int col)
{
int x = 0;
int y = 0;
int win = 0; //用来标记是否取得胜利
int* pw = &win;
while ((row * col - MINE)>win) //row*col-MINE就是全部的非雷位置
{
printf("请输入你想要排查的坐标->");
scanf("%d %d", &x, &y);
if (x >= 1 && x <= row && y >= 1 && y <= col) //判断坐标合法性值得注意的是我们
{ //仍是在11*11的数组中进行操作
if (mine[x][y] == '1')
{
system("cls"); //将之前的记录删除
printf("很遗憾,你被炸死了!\n");
BoardPrint(mine, row, col); //让用户知道自己怎么死的
break;
}
else
{
if (show[x][y] != '*') //判断是否重复排查
{
printf("该坐标已被排查,请重新输入!\n");
continue; //直接进入下一次循环
}
else
{
ExplosionSpread(mine, show, row, col, x, y, pw); //爆炸展开一片
system("cls"); //清空屏幕,将之前的过程全部清除
BoardPrint(show, row, col); //打印棋盘,展示结果
}
}
}
else
{
printf("输入错误,请重新输入!\n");
}
}
if (win == row * col - MINE)
{
system("cls");
printf("恭喜你,排雷成功!\n");
BoardPrint(show, row, col);
return;
}
}
统计一个位置周围八个的含雷数的函数
//获取坐标周围雷的个数
int GetMineCount(char board[ROWS][COLS], int x, int y)
{
int i = 0;
int j = 0;
int count = 0;
for (i = x - 1; i <= x + 1; i++)
{
for (j = y - 1; j <= y + 1; j++)
{
if (board[i][j] == '1')
{
count++;
}
}
}
return count;
}
扩散展开——这里我们需要传递的参数较多,我们在展开前还需一个函数,就是统计该位置周围八个位置的雷的总数,只有当这个位置周围八个全为空才可以释放,否则雷就会暴露出来
//使用递归的方式展开
void ExplosionSpread(char mine[ROWS][COLS], char show[ROWS][COLS], int row, int col, int x, int y, int* pw)
{
if (x >= 1 && x <= row && y >= 1 && y <= col) //判断坐标是否为排查范围内
{
int num = GetMineCount(mine, x, y); //获取坐标周围雷的个数
if (num == 0)
{
(*pw)++; //每递归一次说明有一个空白位被处理
show[x][y] = ' '; //如果该坐标周围没有雷,就把该坐标置成空格,并向周围八个坐标展开
int i = 0;
int j = 0;
for (i = x - 1; i <= x + 1; i++)
{
for (j = y - 1; j <= y + 1; j++) //这两层循环可以产生周围八个坐标
{
if (show[i][j] == '*') //限制递归条件,防止已经排查过的坐标再次递归,从而造成死递归
ExplosionSpread(mine, show, row, col, i, j, pw);//直到遇到有雷了才停下来
}
}
}
else
{
(*pw)++;
show[x][y] = num + '0';
}
}
}
到这我们整个扫雷游戏就做完了,是不是思路清晰了不少呢,那么赶紧自己动手敲一下吧,觉得还不错的话给个免费的赞吧。