C语言小游戏:扫雷实现

目录

扫雷的介绍

部分效果展示

 

游戏实现 

开始准备

开始界面

 初始化棋盘

打印棋盘

随机生成雷

排雷过程

排雷函数主体

信息函数

计算周围雷数的函数

 雷区标记函数

判断输赢

 完整代码实现


扫雷的介绍

扫雷就是要把所有非地雷的格子揭开即胜利;踩到地雷格子就算失败。 游戏主区域由很多个方格组成。 使用鼠标左键随机点击一个方格,方格即被打开并显示出方格中的数字;方格中数字则表示其周围的8个方格隐藏了几颗雷。

这里我们用c语言模拟扫雷游戏,当然,由于作者水平有限,暂时不能做出图形,也暂时不能使用鼠标操作,只能用键盘操作

部分效果展示

 

游戏实现 

开始准备

开始界面

我们可以首先打印菜单,提醒玩家是否开始游戏,通过while循环控制,可以达到一次玩完不过瘾还能接着玩的目的

int main()
{
	int input = 0;
	do
	{
		menu();
		printf("请选择:>");
		scanf("%d", &input);
		switch (input)
		{
		case 1:
			printf("游戏开始!\n");
			game();//封装的游戏主体,后文介绍
			break;
		case 0:
			printf("退出游戏\n");
			system("pause");
			break;
		default:
			printf("选择错误,重新选择!\n");
			break;
		}
	} while (input);
	return 0;
}

菜单函数如下

void menu()
{
	printf("*********************************\n");
	printf("*****欢迎游玩扫雷游戏!**********\n");
	printf("*****made by 东条希尔薇**********\n");
	printf("*******1.     play***************\n");
	printf("*******0.     exit***************\n");
	printf("*********************************\n");

}

打印效果如下

 初始化棋盘

我们在这里需要定义两个二维数组,一个数组用于展示给玩家,并储存排雷信息,一个数组用于在后台随机生成雷并储存。用两个数组储存可以有效的防止排雷过程中产生的歧义(例如,把雷设为‘1’,那么就不清楚‘1’到底是给玩家的提示还是放置的雷)。

当然,我们最好将雷设为‘1’,至于为什么这样设置,见下文

void make_board(char board[ROWS][COLS], int rows, int cols, char set)
{
	for (int i = 0; i < rows; i++)
	{
		for (int j = 0; j < cols; j++)
		{
			board[i][j] = set;
		}
	}
}

在初始化的时候

	char show[ROWS][COLS];//ROWS已经在#define中定义为11,ROW,已定义为9
	char mine[ROWS][COLS];
	make_board(show, ROWS, COLS, '*');//将*设置为不知道信息的区域
	make_board(mine, ROWS, COLS, '0');//0来初始化,设为非雷区

我们模拟的是简单模式的9*9,那为什么要把数组初始化为11*11的呢?

为了防止数组越界

画图示意

打印棋盘

我们只需要打印中间的9*9区域即可,而且只需要打印show数组,存储雷的数组不需要显示给玩家

void display_board(char board[ROWS][COLS], int rows, int cols)
{
	int i = 0;
	int j = 0;
	for (i = 0; i < rows-1; i++)//打印行数,方便玩家操作
	{
		printf("%d ", i);
	}
	printf("\n");
	for (i = 1; i < rows - 1; i++)
	{
		printf("%d ", i);//打印列数
		for (j = 1; j < cols - 1; j++)
		{
			printf("%c ", board[i][j]);
		}
		printf("\n");
	}
}

效果如下

随机生成雷

我们可以使用stdlib.h中的rand函数来随机生成雷的坐标,为了能保证每次游戏的随机生成结果不同,可以在main函数中使用srand和time函数。

void set_mine(char board[ROWS][COLS], int row, int col)
{
	int count = COUNT;//COUNT已用#define定义,方便后期维护,使用简单难道的雷数
	while (count)
	{
		int x = rand() % row + 1;
		int y = rand() % col + 1;
		if (board[x][y] != '1')
		{
			board[x][y] = '1';
			count--;//只有在坐标没被占用时才会放置,防止最后生成的雷数小于10个
		}
	}
}

准备工作分界线


排雷过程

排雷函数主体

需要玩家选择排雷的坐标,并将该坐标周围的信息反馈给玩家,当然,玩家可能会不慎选择不合法的坐标,所以可以通过循环让玩家反复选择,直到选择到合法的坐标为止

void find_mine(char show[ROWS][COLS], char mine[ROWS][COLS], int row, int col)
{
	
	int x = 0;
	int y = 0;
	int count1 = 0;
	while (1)
	{
		printf("请选择要排的坐标\n");
		scanf("%d%d", &x, &y);
		if (x >= 1 && x <= row && y >= 1 && y <= col)//合法的坐标范围
		{
			if (mine[x][y] == '1')
			{
				printf("很遗憾,你被炸死了!\n");//玩家踩到雷了
				display_board(mine, ROWS, COLS);
				system("pause");
				break;
			}
			else
			{
				if (show[x][y] != '*')
				{
					printf("坐标被占用!重新输入\n");//玩家输入的坐标已经被开过了
				}
				else
				{
					open_board(mine,show, ROW, COL, x, y);//后文会提到的信息函数
					display_board(show, ROWS, COLS);//将每次更新的信息呈现给玩家
					printf("你是否需要标记?1是 or 0否\n");//后文将会提到的标记函数
					int ret = 0;
					scanf("%d", &ret);
					if (ret == 1)
					{
						printf("开始标记\n");
						count1 = flag(mine, show, row, col);
						if (count1 == COUNT)
						{
							printf("恭喜,你赢了!\n");
							break;
						}
					}
						
					else
						printf("继续\n");
					
				}
			}
		}
		else
		{
			printf("坐标非法!重新输入!\n");//输入坐标超出棋盘范围
		}

	}
	
}

信息函数

这里我们使用了函数的递归,可以实现以下的效果,能有效防止反复操作的问题

 

void open_board(char mine[ROWS][COLS], char show[ROWS][COLS],int row,int col,int x,int y)
{
	if (x >= 1 && x <= row && y >= 1 && y <= col)
	{
		if (show[x][y] == ' ')
		{
			return;//递归的停止条件
		}
		else if (count_mine(mine, x, y) != 0)//判断周围是否有雷,不用直接遍历判断,有就呈现信息
		{
			show[x][y] = count_mine(mine, x, y) + '0';//后文提到的计算雷数的函数
		}
		else
		{
			show[x][y] = ' ';//标记,防止重复调用,造成递归死循环
			for (int i = x - 1; i <= x + 1; i++)
			{
				for (int j = y - 1; j <= y + 1; j++)
				{
					
					open_board(mine, show, row, col, i, j);//查看以该坐标中心的周围8格区域
				}
			}
		}
	}
	

}

计算周围雷数的函数

这里将‘1’设为雷的优越性就体现出来了。

可以直接把‘1’减去字符0,将其转化为数字相加,在呈现信息的时候再加上‘0’转化为字符。

int count_mine(char mine[ROWS][COLS], int x, int y)
{
	int count = 0;
	for (int i = x - 1; i <= x + 1; i++)
	{
		for (int j = y - 1; j <= y + 1; j++)
		{
			count += (mine[i][j] - '0');
		}
	}
	return count;
	
}

呈上效果图

 雷区标记函数

我们可以模拟扫雷游戏中的标记功能,供玩家标记他们认为有可能有雷的坐标,当然,如果标记错了,可以随时取消

int flag(char mine[ROWS][COLS], char show[ROWS][COLS], int row, int col)
{
	int input = 0;
	int count = 0;
	int x = 0;
	int y = 0;
	do
	{
		printf("请标记你认为的雷的位置,输入已经标记的坐标已取消\n");
		scanf("%d%d", &x, &y);
		if (x >= 1 && x <= row && y >= 1 && y <= col)//判断坐标是否合法
		{
			if (show[x][y] == '*')
			{
				show[x][y] = '$';//用$作为旗帜标记
				for (int i = 1; i <= row; i++)
				{
					for (int j = 1; j <= col; j++)
					{
						if (show[i][j] == '$' && mine[i][j] == '1')
						{
							count++;//下文提到的判断输赢的方式
						}
					}
				}
				display_board(show, ROWS, COLS);//每次标记完后打印棋盘
				
			}
			else if (show[x][y] == '$')
			{
				show[x][y] = '*';//实现取消标记的功能
			}
			else
			{
				printf("坐标被占用!\n");//坐标已经被开过或者已经被标记过
			}

		}
		else
		{
			printf("坐标非法,重新输入!\n");
		}
		printf("是否继续?1是 or 0否\n");
		scanf("%d", &input);
	} while (input);//可以让玩家一直标记,直到玩家主动停止标记
	return count;
}

判断输赢

我们上文中的flag函数有个返回值,返回show数组中的旗帜$和mine数组中的雷重合的个数,判断,如果重合个数等于了COUNT就判断赢了,如果踩到雷,就判断输了

count1 = flag(mine, show, row, col);
						if (count1 == COUNT)
						{
							printf("恭喜,你赢了!\n");
							break;
						}
if (mine[x][y] == '1')
			{
				printf("很遗憾,你被炸死了!\n");
				display_board(mine, ROWS, COLS);
				system("pause");
				break;
			}

为了验证我们逻辑的正确性,将雷数设置为1,来测试一下我们的小游戏

 完整代码实现

main.c文件

#include"game.h"

void game()
{
	char show[ROWS][COLS];
	char mine[ROWS][COLS];
	make_board(show, ROWS, COLS, '*');
	make_board(mine, ROWS, COLS, '0');
	set_mine(mine, ROW, COL);
	display_board(show, ROWS, COLS);
	//display_board(mine, ROWS, COLS);

	find_mine(show, mine, ROW, COL);

}


int main()
{
	srand((unsigned int)time(NULL));
	int input = 0;
	do
	{
		menu();
		printf("请选择:>");
		scanf("%d", &input);
		switch (input)
		{
		case 1:
			printf("游戏开始!\n");
			game();
			break;
		case 0:
			printf("退出游戏\n");
			system("pause");
			break;
		default:
			printf("选择错误,重新选择!\n");
			break;
		}
	} while (input);
	return 0;
}

game.c文件

#include"game.h"

void menu()
{
	printf("*********************************\n");
	printf("*****欢迎游玩扫雷游戏!**********\n");
	printf("*****made by 东条希尔薇**********\n");
	printf("*******1.     play***************\n");
	printf("*******0.     exit***************\n");
	printf("*********************************\n");

}

void make_board(char board[ROWS][COLS], int rows, int cols, char set)
{
	for (int i = 0; i < rows; i++)
	{
		for (int j = 0; j < cols; j++)
		{
			board[i][j] = set;
		}
	}
}

void display_board(char board[ROWS][COLS], int rows, int cols)
{
	int i = 0;
	int j = 0;
	for (i = 0; i < rows-1; i++)
	{
		printf("%d ", i);
	}
	printf("\n");
	for (i = 1; i < rows - 1; i++)
	{
		printf("%d ", i);
		for (j = 1; j < cols - 1; j++)
		{
			printf("%c ", board[i][j]);
		}
		printf("\n");
	}
}

void set_mine(char board[ROWS][COLS], int row, int col)
{
	int count = COUNT;
	while (count)
	{
		int x = rand() % row + 1;
		int y = rand() % col + 1;
		if (board[x][y] != '1')
		{
			board[x][y] = '1';
			count--;
		}
	}
}

void find_mine(char show[ROWS][COLS], char mine[ROWS][COLS], int row, int col)
{
	
	int x = 0;
	int y = 0;
	int count1 = 0;
	while (1)
	{
		printf("请选择要排的坐标\n");
		scanf("%d%d", &x, &y);
		if (x >= 1 && x <= row && y >= 1 && y <= col)
		{
			if (mine[x][y] == '1')
			{
				printf("很遗憾,你被炸死了!\n");
				display_board(mine, ROWS, COLS);
				system("pause");
				break;
			}
			else
			{
				if (show[x][y] != '*')
				{
					printf("坐标被占用!重新输入\n");
				}
				else
				{
					open_board(mine,show, ROW, COL, x, y);
					display_board(show, ROWS, COLS);
					printf("你是否需要标记?1是 or 0否\n");
					int ret = 0;
					scanf("%d", &ret);
					if (ret == 1)
					{
						printf("开始标记\n");
						count1 = flag(mine, show, row, col);
						if (count1 == COUNT)
						{
							printf("恭喜,你赢了!\n");
							break;
						}
					}
						
					else
						printf("继续\n");
					
				}
			}
		}
		else
		{
			printf("坐标非法!重新输入!\n");
		}

	}
	
}

int count_mine(char mine[ROWS][COLS], int x, int y)
{
	int count = 0;
	for (int i = x - 1; i <= x + 1; i++)
	{
		for (int j = y - 1; j <= y + 1; j++)
		{
			count += (mine[i][j] - '0');
		}
	}
	return count;
	
}

int flag(char mine[ROWS][COLS], char show[ROWS][COLS], int row, int col)
{
	int input = 0;
	int count = 0;
	int x = 0;
	int y = 0;
	do
	{
		printf("请标记你认为的雷的位置,输入已经标记的坐标已取消\n");
		scanf("%d%d", &x, &y);
		if (x >= 1 && x <= row && y >= 1 && y <= col)
		{
			if (show[x][y] == '*')
			{
				show[x][y] = '$';
				for (int i = 1; i <= row; i++)
				{
					for (int j = 1; j <= col; j++)
					{
						if (show[i][j] == '$' && mine[i][j] == '1')
						{
							count++;
						}
					}
				}
				display_board(show, ROWS, COLS);
				
			}
			else if (show[x][y] == '$')
			{
				show[x][y] = '*';
			}
			else
			{
				printf("坐标被占用!\n");
			}

		}
		else
		{
			printf("坐标非法,重新输入!\n");
		}
		printf("是否继续?1是 or 0否\n");
		scanf("%d", &input);
	} while (input);
	return count;
}

void open_board(char mine[ROWS][COLS], char show[ROWS][COLS],int row,int col,int x,int y)
{
	if (x >= 1 && x <= row && y >= 1 && y <= col)
	{
		if (show[x][y] == ' ')
		{
			return;
		}
		else if (count_mine(mine, x, y) != 0)//判断周围是否有雷,不用直接遍历判断
		{
			show[x][y] = count_mine(mine, x, y) + '0';
		}
		else
		{
			show[x][y] = ' ';//标记,防止重复调用
			for (int i = x - 1; i <= x + 1; i++)
			{
				for (int j = y - 1; j <= y + 1; j++)
				{
					
					open_board(mine, show, row, col, i, j);
				}
			}
		}
	}
	

}



game.h文件

#pragma once
#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
#include<Windows.h>
#include<stdlib.h>
#include<time.h>
#define ROW 9
#define COL 9
#define ROWS ROW+2
#define COLS COL+2
#define COUNT 10
void menu();
void make_board(char board[ROWS][COLS], int rows, int cols, char set);
void display_board(char board[ROWS][COLS], int rows, int cols);
void set_mine(char board[ROWS][COLS], int row, int col);
void find_mine(char show[ROWS][COLS], char mine[ROWS][COLS], int row, int col);
int count_mine(char mine[ROWS][COLS], int x, int y);
void open_board(char mine[ROWS][COLS], char show[ROWS][COLS], int row, int col,int x,int y);
int flag(char mine[ROWS][COLS], char show[ROWS][COLS], int row, int col);

由于作者水平有限,代码中如有任何的bug,不足之处在所难免!希望各位大佬提出你们宝贵的意见,笔芯~

  • 31
    点赞
  • 26
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 8
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 8
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

东条希尔薇

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值