从零开始,使用C语言实现扫雷小游戏

news2025/1/16 21:04:17

扫雷

    • 1. 前言
    • 2. 准备工作
    • 3. 设计思路
    • 4. 定义数组
    • 5. 初始化
    • 6. 打印
    • 7. 布置雷
    • 8. 排查雷
    • 9. 完整代码

在这里插入图片描述

1. 前言

大家好,我是努力学习游泳的鱼。今天我们会用C语言实现一个经典的windows小游戏:扫雷。扫雷是一款单机小游戏,我上中学时特喜欢在电脑课上玩,研究应对各种情况的思路,每次通关最高难度的关卡都会开心好一阵。现在学会了C语言,总算可以自己实现扫雷了。话不多说,咱们开始吧。
在这里插入图片描述

2. 准备工作

我们新建一个项目,并创建三个文件:

  1. test.c - 负责测试游戏代码。
  2. game.c - 负责游戏功能的具体实现。
  3. game.h - 负责头文件的包含,符号的定义,函数的声明。

test.cgame.c里都要#include "game.h"
测试游戏时,玩一把肯定不过瘾,会想要再来一把,这就需要用到do while循环。

void menu()
{
	printf("****************************\n");
	printf("********  1. play    *******\n");
	printf("********  0. exit    *******\n");
	printf("****************************\n");
}

int main()
{
	int input = 0;

	do
	{
		menu();
		printf("请选择(1/0):>");
		scanf("%d", &input);
		switch (input)
		{
		case 1:
			game();
			break;
		case 0:
			printf("退出游戏\n");
			break;
		default:
			printf("选择错误,重新选择!\n");
			break;
		}
	} while (input);

	return 0;
}

3. 设计思路

接下来讲解扫雷游戏(game函数)实现的思路。

  1. 布置雷 - 10个。
  2. 扫雷 - 玩法如下:
    输入坐标,此时分两种情况:
    1. 是雷 – 就被炸了,游戏结束。
    2. 不是雷 – 就告诉你这个坐标周围8个坐标上总共有多少个雷。

直到把所有非雷的位置全部都找出来,游戏结束,扫雷成功。
我们假设扫雷的棋盘是9×9的。我们布置雷的信息要想全部存起来,就需要使用9×9的二维数组。
怎么布置雷呢?假设要布置10个雷,我们就随机生成10个坐标,把数组的这10个位置都置成1,其余位置存储0。实际排查的时候,只需要显示周围8个坐标有几个1就行了。
但是这样设计有一个问题,假设有一个位置周围只有1个雷,那就显示1。我们如何判断这个1是表示雷的1,还是显示周围有1个雷的1呢?这就有歧义了。
如何解决这个问题呢?我们可以再搞一个一样大的数组。两个数组,一个放布置好的雷的信息,另一个放排查出的雷的信息。对于后者,如果某个位置没有排查过,就存储*,以保持神秘感;如果排查过了,并且不是雷,就存储周围雷的个数。由于*是一个字符,为了保持类型的统一,雷的个数也要用数字字符来存储(如某位置周围有3个雷,就存储字符3),那存储排查出的雷的信息的数组就是一个9×9char类型的数组。还是为了保持类型的统一,存储雷的信息的数组中,我们用字符0表示非雷,字符1表示雷,该数组也是一个9×9char类型的数组。
阶段总结一下:

  1. char mine[9][9]负责存储布置好的雷的信息,字符1表示雷,字符0表示非雷。
  2. char show[9][9]负责存储排查出的雷的信息,*表示未排查,数字字符表示已排查。

但是,这样设计仍然有问题。如果我们要排查数组边上或角上的位置,我们需要访问该位置周围的8个位置,就有可能越界访问了。
如何解决这个问题呢?我们可以把存放雷的信息的数组开大一圈,这样访问时,排查边上或角上的数据就不会越界了。为了保持两个数组的一一对应,另一个数组也开大一圈。经过以上的分析,如果实际使用的棋盘大小是9×9的,两个数组就应该定义为11×11的。

4. 定义数组

为了以后修改这点方便,我们定义几个宏,ROWCOL为实际使用的大小(9×9),ROWSCOLS为实际定义数组的大小(11×11)。

#define ROW 9
#define COL 9

#define ROWS (ROW + 2)
#define COLS (COL + 2)

接着定义两个数组。

char mine[ROWS][COLS] = { 0 };
char show[ROWS][COLS] = { 0 };

5. 初始化

mine数组由于要存储雷的信息,没布置雷前,我们想把这个数组全部初始化成0show数组用于存放排查出来的雷的信息,一开始应全部初始化成*。所以我们需要一个初始化函数。由于两个数组初始化的内容不一样,所以我们设计函数时,需要把初始化的内容当做参数传过去。

init_board(mine, ROWS, COLS, '0');
init_board(show, ROWS, COLS, '*');

具体的实现,只需遍历数组就行了。

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

6. 打印

对两个数组进行初始化后,我们想把它们打印出来看看。
初始化时,我们需要初始化整个数组(11×11),但是打印以及后面的操作,我们基本只关心中间的9×9,周围的一圈只是为了防止越界。

show_board(mine, ROW, COL);
show_board(show, ROW, COL);

具体的实现,也是遍历数组除去最外面一圈的元素,那下标应从1开始,最大是rowcol

void show_board(char arr[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 ", arr[i][j]);
		}
		printf("\n");
	}
}

打印出来效果如下:
在这里插入图片描述
但是这样打印不够完美,我们每次还要去数某个位置是第几行第几列,所以最好把行标和列标也打印出来。
每次打印一行前,我们都打印下行号printf("%d ", i);
在所有信息打印前,我们把列标打印出来。

for (i = 0; i <= col; ++i)
{
	printf("%d ", i);
}
printf("\n");

当然,我们可以在打印的最前和最后加上分割行。printf("------------扫雷------------\n");
下面是打印函数完整的代码。

void show_board(char arr[ROWS][COLS], int row, int col)
{
	int i = 0;
	printf("------------扫雷------------\n");
	for (i = 0; i <= col; ++i)
	{
		printf("%d ", i);
	}
	printf("\n");

	for (i = 1; i <= row; ++i)
	{
		printf("%d ", i);
		int j = 0;
		for (j = 1; j <= col; ++j)
		{
			printf("%c ", arr[i][j]);
		}
		printf("\n");
	}
	printf("------------扫雷------------\n");
}

打印效果如下:
在这里插入图片描述

7. 布置雷

我们需要在mine数组里布置雷。set_mine(mine, ROW, COL);
假设雷的个数是EASY_COUNT#define EASY_COUNT 10
具体的实现,我们需要写一个循环,每次随机生成一个坐标,如果这个位置不是雷,就在这个位置放雷,知道把所有的雷放完为止。

void set_mine(char mine[ROWS][COLS], int row, int col)
{
	int count = EASY_COUNT;
	int x = 0;
	int y = 0;
	while (count)
	{
		x = rand() % row + 1;
		y = rand() % col + 1;
		if ('0' == mine[x][y])
		{
			mine[x][y] = '1'; // 布置雷
			--count;
		}
	}
}

不要忘记在调用rand函数之前要调用srand函数,并给srand函数传递用time函数生成的时间戳。srand((unsigned int)time(NULL));rand函数和srand函数需要引用头文件stdlib.htime函数需要引用头文件time.h
我们可以把生成雷的信息打印出来。show_board(mine, ROW, COL);
在这里插入图片描述

8. 排查雷

布置好雷后,就开始排查雷。排查雷需要同时操作两个数组。find_mine(mine, show, ROW, COL);
排查雷时,可以通过一个循环反复获取坐标,并对坐标进行判断。

  1. 首先判断坐标的合法性,横纵坐标都必须在1row(col)之间。
  2. 如果坐标合法,再看这个坐标有没有排查过,如果show数组在该位置还是*,说明没有排查过。
  3. 如果坐标还没排查过,再看是不是雷,是雷的话游戏结束,不是雷的话就显示该坐标周围有几个雷。

循环会在两种情况下结束,

  1. 一种是踩到雷了,直接break出去,
  2. 另一种是找到所有非雷的位置,就排雷成功了。总共有row×col个位置,一共有EASY_COUNT个雷,那非雷的位置个数就是两者相减。判断是否排雷成功,可以在循环条件中判断。
void find_mine(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);
		if (x >= 1 && x <= row && y >= 1 && y <= col)
		{
			if ('*' == show[x][y])
			{
				if ('1' == mine[x][y])
				{
					printf("很遗憾,你被炸死了\n");
					show_board(mine, row, col);
					break;
				}
				else
				{
					int count = get_mine_count(mine, x, y);
					show[x][y] = count + '0';
					show_board(show, row, col);
					++win;
				}
			}
			else
			{
				printf("该坐标已被排查\n");
			}
		}
		else
		{
			printf("坐标非法,重新输入\n");
		}
	}

	if (row * col - EASY_COUNT == win)
	{
		printf("恭喜你,排雷成功\n");
		show_board(mine, row, col);
	}
}

我们用get_mine_count函数来获取某个位置(坐标为x,y)周围8个坐标雷的个数。由于字符1的ASCII码值减去字符0的ASCII码值是1,而mine数组里存放的就是字符1和字符0。所以我们只需要把mine数组中,该位置周围八个坐标存储的字符加起来,再减去字符0的ASCII码值的八倍,就能算出一共有多少个雷了。由于get_mine_count函数只在find_mine函数中使用,所以加上static

static int get_mine_count(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';
}

到此为止,整个扫雷游戏就写完啦。

9. 完整代码

game.h

#pragma once

#include <stdio.h>
#include <stdlib.h>
#include <time.h>

#define ROW 9
#define COL 9

#define ROWS (ROW + 2)
#define COLS (COL + 2)

#define EASY_COUNT 10

// 初始化
void init_board(char arr[ROWS][COLS], int rows, int cols, char set);

// 打印
void show_board(char arr[ROWS][COLS], int row, int col);

// 布置雷
void set_mine(char mine[ROWS][COLS], int row, int col);

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

game.c

#define _CRT_SECURE_NO_WARNINGS 1

#include "game.h"

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

void show_board(char arr[ROWS][COLS], int row, int col)
{
	int i = 0;
	printf("------------扫雷------------\n");
	for (i = 0; i <= col; ++i)
	{
		printf("%d ", i);
	}
	printf("\n");

	for (i = 1; i <= row; ++i)
	{
		printf("%d ", i);
		int j = 0;
		for (j = 1; j <= col; ++j)
		{
			printf("%c ", arr[i][j]);
		}
		printf("\n");
	}
	printf("------------扫雷------------\n");
}

void set_mine(char mine[ROWS][COLS], int row, int col)
{
	int count = EASY_COUNT;
	int x = 0;
	int y = 0;
	while (count)
	{
		x = rand() % row + 1;
		y = rand() % col + 1;
		if ('0' == mine[x][y])
		{
			mine[x][y] = '1'; // 布置雷
			--count;
		}
	}
}

static int get_mine_count(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 find_mine(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);
		if (x >= 1 && x <= row && y >= 1 && y <= col)
		{
			if ('*' == show[x][y])
			{
				if ('1' == mine[x][y])
				{
					printf("很遗憾,你被炸死了\n");
					show_board(mine, row, col);
					break;
				}
				else
				{
					int count = get_mine_count(mine, x, y);
					show[x][y] = count + '0';
					show_board(show, row, col);
					++win;
				}
			}
			else
			{
				printf("该坐标已被排查\n");
			}
		}
		else
		{
			printf("坐标非法,重新输入\n");
		}
	}

	if (row * col - EASY_COUNT == win)
	{
		printf("恭喜你,排雷成功\n");
		show_board(mine, row, col);
	}
}

test.c

#define _CRT_SECURE_NO_WARNINGS 1

#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] = { 0 };

	// 初始化棋盘
	init_board(mine, ROWS, COLS, '0');
	init_board(show, ROWS, COLS, '*');
	// 打印棋盘
	//show_board(mine, ROW, COL);
	// 布置雷
	set_mine(mine, ROW, COL);
	show_board(show, ROW, COL);
	// 排查雷
	find_mine(mine, show, ROW, COL);
}

int main()
{
	int input = 0;
	srand((unsigned int)time(NULL));

	do
	{
		menu();
		printf("请选择(1/0):>");
		scanf("%d", &input);
		switch (input)
		{
		case 1:
			game();
			break;
		case 0:
			printf("退出游戏\n");
			break;
		default:
			printf("选择错误,重新选择!\n");
			break;
		}
	} while (input);

	return 0;
}

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

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

相关文章

网易24届内推

【网易】2024届网易互联网秋季校园招聘内推开始啦&#xff01;给你分享我的专属内推邀请函&#xff1a;https://bole.campus.163.com/campus/home?projectId55&type99&isShare1&boleId7b842acc7c2b42db&boleType2&signatured5f2a3dc23bed70777a8be1a14b49…

简单了解IPv4编址

目录 一、IPv4地址 二、进制转换 三、有类IPv4 四、无类IPv4 3.1 子网掩码 3.2 地址规划 3.3 VLSM可变长子网掩码 五、私有IPv4地址 六、IPv4报文格式 七、IP地址解析 一、IPv4地址 IPv4地址由“网络位主机位”构成&#xff0c;所谓的网络位就是我们通常所指的网段区…

python 列表extend方法和+(拼接)的区别

1.extend方法会直接作用于原始列表&#xff0c;会修改原始列表的值 a [10, 20, 30] b [1, 2, 3]print(a b) print(a) a.extend(b) print(a) 2.extend接收到的参数是一个可迭代的对象(iterable)&#xff0c;不管是list,tuple,str,dict,set a [10, 20, 30] b {"name&qu…

公司内部网段多管控乱,该如何规范跨网文件传输交换?

古往今来&#xff0c;高筑墙一直是有效的防御措施。从边塞长城到护城河外的高高城墙&#xff0c;都是利用隔离地域的形式实现保护安全域的效果。这样一来&#xff0c;城内的安全域可以在遇到危险时受到有效保护。 在企业网络安全防护方面&#xff0c;网络安全域隔离也是网络安全…

3D点云处理:圆柱侧面点云展开为平面 凹凸缺陷检测(附源码)

文章目录 1. 基本内容展开部分推导2. 展开流程3. 代码实现4. 应用文章目录:3D视觉个人学习目录微信:dhlddxB站: Non-Stop_目标:对采集的圆柱面点云展开为平面;应用:可用于检测圆柱侧面的凹凸缺陷;1. 基本内容 圆柱的侧面展开原理是将一个圆柱体(或柱体)的侧面展开成一个…

el-table实现纯前端导出(适用于el-table任意表格)

2023.9.1今天我学习了如何使用el-table实现前端的导出功能&#xff0c;该方法的好处有无论你的el-table长什么样子&#xff0c;导出之后就是什么样子。 1.安装三个插件 npm install file-save npm install xlsx npm install xlx-style 2.创建Export2Excel.js // 根据dom导出表…

Windows右键添加用 IDEA 打开

1.安装IDEA时 安装时会有个选项来添加&#xff0c;如下&#xff1a; 勾选即可 2.修改注册表 安装时未勾选&#xff0c;可以把下面代码中程序路径改为自己的&#xff0c;保存为对应的 idea.reg文件&#xff0c;双击即可 Windows Registry Editor Version 5.00[HKEY_CLASSES…

问题记录:jenkins添加节点时Launch method没有Launch agents via SSH选项

jenkins问题记录 在jenkins主页&#xff0c;左侧点击Manage Jenkins&#xff0c;找到plugins选项&#xff0c;搜索如下插件安装&#xff1a; 安装完插件后&#xff0c;即可看到ssh选项出来了

2023开学礼《乡村振兴战略下传统村落文化旅游设计》西学图灵许少辉八一新书

2023开学礼《乡村振兴战略下传统村落文化旅游设计》西学图灵许少辉八一新书

Linux工具

一、yum yum可以看作一个客户端&#xff08;应用商店&#xff09;、应用程序&#xff0c;它如何知道去哪里下载软件&#xff1f; yum也是一个指令/程序&#xff0c;可以找到它的安装路径。 在list中可以看到yum能安装的所有软件&#xff0c;通过管道找到想要的&#xff0c;yum …

ChatGPT 总结前端HTML, JS, Echarts都包含哪些内容

AIGC ChatGPT ,BI商业智能, 可视化Tableau, PowerBI, FineReport, 数据库Mysql Oracle, Office, Python ,ETL Excel 2021 实操,函数,图表,大屏可视化 案例实战 http://t.csdn.cn/zBytu

数据视觉化探秘:了解有效传达的几种常见图表

当涉及向观众传达复杂的数据时&#xff0c;数据可视化图表成为了无价的工具。本文为大家介绍几种常用的数据可视化图表&#xff0c;以便更好地展示和理解数据。 折线图 这是最基础的图表类型之一&#xff0c;通过连续的折线连接数据点&#xff0c;呈现数据随时间或顺序的变化…

极米投影仪好用吗?极米RS Pro 3的真实体验效果怎么样?

随着智能投影仪的不断普及&#xff0c;现在大多数的年轻家庭在布局客厅的时候也不再以传统的电视为主&#xff0c;投影仪代替电视已然成为了一种常态&#xff0c;投影仪的体积更小、不用固定安装、也不会占用太大的空间&#xff0c;并且可以随意投射出百寸以上的画面大小&#…

《论文阅读21》Equivariant Multi-View Networks

一、论文 研究领域&#xff1a;计算机视觉 | 多视角数据处理中实现等变性论文&#xff1a;Equivariant Multi-View Networks ICCV 2019 论文链接视频链接 二、论文简述 在计算机视觉中&#xff0c;模型在不同视角下对数据&#xff08;例如&#xff0c;点云、图像等&#xff0…

XXE漏洞利用技巧(由简入深)-----portswigger(XXE部分WP)

什么是XXE XXE&#xff08;XML External Entity&#xff1a;xml外部实体注入&#xff09;&#xff0c;它出现在使用XML解析器的应用程序中。XXE攻击利用了XML解析器的功能&#xff0c;允许应用程序从外部实体引用加载数据。攻击者可以通过构造恶意的XML实体引用来读取本地文件…

四、MySql数据类型

文章目录 一、数据类型&#xff08;一&#xff09;数据类型分类&#xff08;二&#xff09;数值类型1、tinyint类型2、bit类型3、小数类型&#xff08;1&#xff09; float&#xff08;2&#xff09;decimal 4、字符串类型&#xff08;1&#xff09;char&#xff08;2&#xff…

101序列检测器

本次所做设计&#xff0c;使用数字电路芯片实现的101序列检测器。电路图如下&#xff1a; 主要首先要根据需求画出状态转移方程&#xff0c;然后写出它的逻辑表达式。最后根据所选触发器种类确定电路图。序列由按键控制输入&#xff0c;按键按下&#xff0c;代表输入1 &#xf…

如何使用Python和正则表达式处理XML表单数据

在日常的Web开发中&#xff0c;处理表单数据是一个常见的任务。而XML是一种常用的数据格式&#xff0c;用于在不同的系统之间传递和存储数据。本文通过阐述一个技术问题并给出解答的方式&#xff0c;介绍如何使用Python和正则表达式处理XML表单数据。我们将探讨整体设计、编写思…

交换机和路由器互联,并用ACL进行网段隔离的实例

拓朴如下&#xff1a; vlan10 20 为业务网段&#xff0c;vlan100为管理网段 S1起了vlan10 20&#xff0c;上行接口起了trunk口 S2起了vlan 10 20 100&#xff0c;分别起了vlanif网关&#xff0c;其中 vlanif 100是100.1.1.2/30&#xff0c;下行配置为trunk&#xff0c;起了AC…

Mysql索引、事务、函数

索引&#xff1a; 索引&#xff0c;使用索引可快速访问数据库表中的特定信息。索引是对数据库表中一列或多列的值进行排序的一种结构。 在关系数据库中&#xff0c;索引是一种与表有关的数据库结构&#xff0c;它可以使对应于表的SQL语句执行得更快。索引的作用相当于图书的目录…