C语言:扫雷游戏实现

news2025/1/11 20:39:02

一、扫雷游戏的分析和设计

扫雷游戏想必大家都玩过吧,初级的玩法是在一个9*9的棋盘上找到没有雷的格子,而今天我们就要做的就是9*9扫雷游戏的实现。

1、游戏功能和规则

  • 使用控制台实现经典的扫雷游戏
  • 游戏可以通过菜单实现继续玩或者退出游戏
  • 扫雷的棋盘是9*9的格⼦
  • 默认随机布置10个雷(可随意)
  • 可以排查雷

如果位置不是雷,就显示周围有几个雷

如果位置是雷,就炸死游戏结束

把除10个雷之外的所有非雷都找出来,排雷成功,游戏结束

2、游戏实现思路

  1. 首先,创建一个主函数用于判断是否进行游戏,并设计一个菜单。
  2. 创建用于存放雷的棋盘,并进行棋盘的初始化。
  3. 设计棋盘的打印。
  4. 放入雷,并实现有雷棋盘的打印。
  5. 输入坐标进行排查,并返回排查结果。
  6. 进行循环直到排查成功或被炸死。

在布置雷的过程中我们将雷设置为'1',不是雷设置为'0'; 

二、代码实现

在实现代码前,我们可以用多文件的方式,对我们的函数进行声明和定义,这样可以简化和简洁直观的设计和表达代码。

  • game.h:我们可以在game.h文件中包含我们所有需要用到的头文件,常量的定义和函数的声明,其他文件只需要声明game.h 即可。
  • game.c:写入游戏所需要的功能函数。
  • test.c:代码的整合

1、主函数的实现

在主函数中我们先创建一个input值,用于输入我们的选择,根据 do while 循环判断我们是否进行游戏,并且每次循环都打印菜单,在接着使用switch语句根据玩家输入进行判断,如果输入为1就开始游戏执行game(),如果为0,游戏结束并跳出循环,如果输入错误就继续进行循环直到主动退出。

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

 2、菜单的实现

不需要返回所以用void

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

 3、游戏函数的实现

1、棋盘的设计

在排雷的过程中如果我们选择的位置不是雷就会看它周围的8个坐标中是否有雷,如果有雷就会显示雷的个数。

但如果我们选择的是边界位置,就会超出棋盘的范围,导致越界。

所以我们要将棋盘扩大一圈,这样就不存在越界问题了。

假设我们现在已经布置好了雷(雷用1表示,不是雷用0表示),当我们输入一个坐标的时候,如果这个坐标周围有一个雷,那么我们需要将这个雷的数量信息记录并存储下来,同时打印给玩家,但是我们存放雷的数组里面已经有数据了,如果再存放在里面就会可能产生混淆从而导致棋盘混乱打印困难,所以我们可以设置两个棋盘,一个是雷的棋盘mine,一个是用于放置雷的信息的棋盘show,我们后面会把雷布置到到mine数组里面,再mine数组排查雷之后,将雷的信息存放到show数组里面,然后打印show数组的信息给用户参考排查,这样就不会存在数据混淆了。

同时我们也要在game.h文件里进行常量的定义。

2、棋盘的初始化

在设计好棋盘后,我们就应该对棋盘进行初始化,首先,我们将mine棋盘全初始化为'0',show棋盘全初始化为'*'.

//棋盘的初始化
void InitBoard(char board[ROWS][COLS], int row, int col, char set)
{
	for (int i = 1; i < row; i++)
	{
		for (int j = 1; j < col; j++)
		{
			board[i][j] = set;
		}
	}
}

因为要初始化两个数组,所以我们需要创建一个char set,用于接收'0' 和 '*' 。

3、打印棋盘

因为是扫雷游戏,所以我们只需要打印一个隐藏信息的棋盘即可

我们只需要将show数组,行和列传给函数,并且为了显示行和列数,我们可以先利用for循环打印0-col的一行数,再进入棋盘的打印 ,在行的循环中打印一列数

//棋盘的打印
void DisplayBoard(char board[ROWS][COLS], int row, int col)
{
	int i = 0;
	int j = 0;
	printf("-----------扫雷-------------\n");
	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]);
		}
		printf("\n");
	}
	printf("-----------扫雷-------------\n");
}

 

4、雷的布置

 我们需要在mine数组里布置雷,那我们该如何布置呢?

1.这就需要用到我们的rand()函数了。

rand()函数是随机数生成函数,需要使用stdlib,h文件,但生成的数不是完全的随机数,而是伪随机数,我们需要改变他的种子,才能真正的生成随机数;

 2.这就需要用到我们的srand()函数了

它一般用于初始化随机数生成器,通常是为rand函数设置种子,以便生成不同的随机数序列。所以我们在每次调用rand函数之前应先调用srand函数,以传入不同的种子。而srand函数通过参数seed来设置它的随机生成数,也就是说我们需要种子的种子不是一个固定值,只有这样,rand函数生成的数才是真正的随机数。

 3、这就用到了time()函数了

因为时间一直在变化,所以我们可以用时间来当作srand函数的种子,它需要包含time.h的头文件。

所以我们在主函数里设计了srand函数

但随机数的范围是没法确定的,所以我们需要固定一个范围

用x表示行用y表示列,如果棋盘长度为9,rand()%9就是0-8,+1就是0-9,同理列也是这样

之后我们设置雷的数量, 设为count=EASY_COUNT,在game.h上也定义常量EASY_COUNT 10;

在利用while循环条件设置为count,布置一个雷count--;在循环中,利用if语句将随机生成的下标位置如果不等与'1',就设置为'1'

//布置雷
void SetMine(char board[ROWS][COLS], int row, int col)
{
	int count = EASY_COUNT;
	while (count)
	{
		//生成随机下标
		//x:1-9
		//y:1-9
		int x = rand() % row + 1;
		int y = rand() % col + 1;
		if (board[x][y] != '1');
		{
			board[x][y] = '1';
			count--;
		}
	}
}

5、雷的排查

排查雷需要利用循环,那么判断循环结束的条件是什么呢? 

把除10个雷之外的所有非雷都找出来,排雷成功,游戏结束

如果位置是雷,就炸死游戏结束

所以条件就是找到不是雷的所有坐标既

设置win为不是雷的坐标数

row*col:全部坐标数

EASY_COUNT:雷数

win<row*col-EASY_COUNT 

我们需要规定输入坐标的范围,如果判断的坐标值为'1',游戏结束,并打印所有雷;如果不是,win++,并返回周围雷数,由于返回的值应该是字符 ,所以要让返回值+'0';并打印所有雷。

如果排查雷数等于win=row*col-EASY_COUNT 就是排查成功,并打印所有雷。

//排查雷
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);
		if (x >= 1 && x <= row && y >= 1 && y <= col)
		{
			if (mine[x][y] == '1')
			{
				printf("很遗憾,此处是雷,被炸死\n");
				DisplayBoard(mine, ROW, COL);
				break;
			}
			else
			{
				win++;
				printf("还要排查%d个位置\n", row * col - EASY_COUNT - win);
				int c = GetMineCount(mine, x, y);
				show[x][y] = c + '0';
				DisplayBoard(show, ROW, COL);
			}
		}
		else
		{
			printf("输入坐标有误,请重新输入\n");
		}

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

6、返回周围雷数

//返回周围雷数
int GetMineCount(char mine[ROWS][COLS], int x, int y)
{
	return mine[x - 1][y - 1] + mine[x - 1][y] + mine[x - 1][y + 1] + mine[x][y - 1] + mine[x][y + 1] 
		+ mine[x + 1][y - 1] + mine[x + 1][y] + mine[x + 1][y + 1]-8*'0';
}

因为雷的字符为1,非雷为0;所以将周围坐标相加就是雷数,再减去8*'0',就是雷的个数.

三、完整代码

game.h

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

#define EASY_COUNT 10

#define ROW 9
#define COL 9

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

void InitBoard(char board[ROWS][COLS],int row, int col, int set);

void DisplayBoard(char board[ROWS][COLS], int row, int col);

void SetMine(char board[ROWS][COLS], int row, int col);

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

test.c

#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 };//用来存放排查出的雷的信息
	//初始化
	InitBoard(mine, ROWS, COLS, '0');
	InitBoard(show, ROWS, COLS, '*');
	//打印棋盘
	//DisplayBoard(mine, ROW, COL);
	DisplayBoard(show, ROW, COL);
	//布置雷
	SetMine(mine, ROW, COL);
	//排查雷
	FindMine(mine,show, ROW, COL);

}

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

game.c

#include"game.h"

//棋盘的初始化
void InitBoard(char board[ROWS][COLS], int row, int col, char set)
{
	for (int i = 1; i < row; i++)
	{
		for (int j = 1; j < col; j++)
		{
			board[i][j] = set;
		}
	}
}

//棋盘的打印
void DisplayBoard(char board[ROWS][COLS], int row, int col)
{
	int i = 0;
	int j = 0;
	printf("-----------扫雷-------------\n");
	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]);
		}
		printf("\n");
	}
	printf("-----------扫雷-------------\n");
}

//布置雷
void SetMine(char board[ROWS][COLS], int row, int col)
{
	int count = EASY_COUNT;
	while (count)
	{
		//生成随机下标
		//x:1-9
		//y:1-9
		int x = rand() % row + 1;
		int y = rand() % col + 1;
		if (board[x][y] != '1');
		{
			board[x][y] = '1';
			count--;
		}
	}
}

//返回周围雷数
int GetMineCount(char mine[ROWS][COLS], int x, int y)
{
	return mine[x - 1][y - 1] + mine[x - 1][y] + mine[x - 1][y + 1] + mine[x][y - 1] + mine[x][y + 1] 
		+ mine[x + 1][y - 1] + mine[x + 1][y] + 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);
		if (x >= 1 && x <= row && y >= 1 && y <= col)
		{
			if (mine[x][y] == '1')
			{
				printf("很遗憾,此处是雷,被炸死\n");
				DisplayBoard(mine, ROW, COL);
				break;
			}
			else
			{
				win++;
				printf("还要排查%d个位置\n", row * col - EASY_COUNT - win);
				int c = GetMineCount(mine, x, y);
				show[x][y] = c + '0';
				DisplayBoard(show, ROW, COL);
			}
		}
		else
		{
			printf("输入坐标有误,请重新输入\n");
		}

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

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

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

相关文章

Git 安装教程

1、登录git 官方网站&#xff1a;https://git-scm.com/ 点击左边的 Downloads 或者 右边标识的下载标志&#xff0c;它根据电脑操作系统自动匹配版本 Downloads for Windows 2、以 windows 为例下载对应版本 网络有时可能不大好&#xff0c;阿里镜像下载超快。 下载好以后&a…

传统物流机械锁控的痛点与难点深度剖析

在当今全球化和高度竞争的商业环境中&#xff0c;物流行业作为经济发展的重要支撑&#xff0c;其高效、安全的运营至关重要。而物流锁控&#xff0c;作为保障货物在运输和存储过程中安全的关键环节&#xff0c;传统机械物流锁控方式却面临着诸多严峻的挑战&#xff0c;这些问题…

Linux系统服务——【web,http协议,apache服务和nginx服务】(sixteen day)

一、web基础以及http协议 1、web基本概念和常识 前端开发一般用uniapp. 1、Web:为用户提供的一种在互联网上浏览信息的服务&#xff0c;Web 服务是动态的、可交互的、跨平台的和图形化的。 2、Web 服务为用户提供各种互联网服务&#xff0c;这些服务包括信息浏览服务&#xf…

关于使用pagehelper分页插件 进行mybatis 一对多嵌套查询时 查询数量太多 无法达到你想展示的效果的问题 --已解决

1.问题 先来描述一下问题 我有一个商品表 里面嵌套 1.1如下是我的数据库 一个商品对应两个口味对应三个商品轮播图&#xff0b;两个商品描述图 那可不是1*2*2*3 12 就是分页里面的pageSize 为12 时才能显示完这个商品的完整数据 这是我的sql语句 resultMap映射 2.正确做…

【香橙派系列教程】(二)刷机和系统启动

&#xff08;二&#xff09;刷机和系统启动 文章目录 &#xff08;二&#xff09;刷机和系统启动1.刷机2.基于 Windows PC 将 Linux 镜像烧写到 TF 卡的方法3.Debian 和 Ubuntu 系统使用说明4.串口登录系统5.修改登陆密码6.修改网络配置7.SSH登录开发板8.修改开发板内核启动日志…

学习008-02-04-05 Make a List View Editable(使列表视图可编辑)

Make a List View Editable&#xff08;使列表视图可编辑&#xff09; This lesson explains how to make a List View editable. 本课介绍如何使列表视图可编辑。 The instructions below show how to create new objects of the DemoTask type directly in the Task List V…

MSPM0G3507之电赛小车

一、前言 本文没什么技术分享&#xff0c;纯聊天。以下内容均为笔者的浅薄理解&#xff0c;有不对的地方还请多多包涵。 二、相关配置 主控单元&#xff1a;MSPM0G3507SPTR&#xff08;48角&#xff09; 编译环境&#xff1a;Keil5.33、5.39&#xff08;推荐&#xff09;都可 …

一分钟小课堂!电脑怎么录屏?4款热门软件实操指南

在这个信息超多的时代里&#xff0c;学会电脑怎么录屏可真是一门挺有用的技能。不管是你想做教学视频、录下游戏里好玩的瞬间&#xff0c;还是展示一下工作上的步骤&#xff0c;掌握录屏的方法都能让你的分享更直观、更有效率。今天&#xff0c;咱们就来聊聊四款很火的录屏大师…

vue3-环境变量-JavaScript-axio-基础使用-lzstring-字符串压缩-python

文章目录 1.Vue3环境变量1.1.简介1.2.全局变量的引用1.3.package.json文件 2.axio2.1.promise2.2.安装2.3.配置2.3.1.全局 axios 默认值2.3.2.响应信息格式 2.4.Axios的拦截器2.4.1.请求拦截器2.4.2.响应拦截器2.4.3.移除拦截器2.4.4.自定义实例添加拦截器 3.lz-string3.1.java…

Tantivy使用Rust 开发的全文搜索引擎库

一、概述 Tantivy是一个全文搜索引擎库&#xff0c;灵感来自Apache Lucene&#xff0c;用Rust编写。 如果你正在寻找Elasticsearch或Apache Solr的替代品&#xff0c;请查看我们基于Tantivy构建的分布式搜索引擎Quiuckwit。 Tantivy更接近Apache Lucene&#xff0c;而不是E…

仅需一分钟,使用极空间部署一个强大的开源问卷考试系统『SurveyKing』

仅需一分钟&#xff0c;使用极空间部署一个强大的开源问卷考试系统『SurveyKing』 哈喽小伙伴们好&#xff0c;我是Stark-C~ 我们生活中估计应该都收到了不少的问卷调查吧&#xff1f;很多商家或者运营商都会通过问卷调查的方式了解客户满意度&#xff0c;或者高市场调研&…

数据中台建设之数据汇聚与数据交换

目录 一、数据汇聚 1.1 概述 1.2 汇聚数据类型 1.2.1 结构化数据 1.2.2 半结构化数据 1.2.3 非结构化数据 1.3 汇聚数据模式 1.3.1 概述 1.3.2 离线 1.3.3 实时 1.4 汇聚数据方法 1.4.1 概述 1.4.2 ETL 1.4.3 ELT 1.5 汇聚数据工具 1.5.1 概述 1.5.2 Flink CDC…

Java人力资源招聘社会校招类型招聘系统PC端

&#x1f50d;【揭秘】人力资源新利器&#xff01;社会校招一站式PC端招聘系统全攻略&#x1f680; &#x1f308; 开篇引言&#xff1a;招聘新纪元&#xff0c;效率为王&#xff01; Hey小伙伴们&#xff0c;你是否还在为繁琐的招聘流程头疼不已&#xff1f;&#x1f92f; 面…

Spark累加器(Accumulator)

1.累加器类型&#xff1a; 数值累加器&#xff1a;用于计算总和、计数等。布尔累加器&#xff1a;用于计算满足特定条件的次数。自定义累加器&#xff1a;允许定义复杂的聚合逻辑和数据结构。集合累加器&#xff1a;用于计算唯一元素的数量&#xff0c;处理去重操作。 在 Spar…

Qt Designer,仿作一个ui界面的练习(四):编写代码

一、新建项目&#xff0c;目录结构如图&#xff1a; PYS下存放脚本&#xff0c;SRC下存放资源文件&#xff0c;UIS下存放组态画面文件。 在每个子目录下都有__init__.py文件&#xff0c;系统会自动将其识别为软件包。 其中一个UIS.__init__.py文件的内容&#xff1a; # impo…

手撕数据结构02--二分搜索(附源码)

一、理论基础 二分搜索&#xff0c;也称折半搜索、对数搜索&#xff0c;是一种在有序数组中查找某一特定元素的搜索算法。 二分搜索是一种高效的查找算法&#xff0c;适用于在已排序的数组中查找特定元素。它的基本思想是通过不断将搜索区间对半分割&#xff0c;从而快速缩小…

ROOM数据快速入门

ROOM数据库快速入门 文章目录 ROOM数据库快速入门第一章 准备工作第01节 引入库第02节 布局文件第03节 activity类第04节 效果图 第二章 数据类第01节 实体类&#xff08;表&#xff09;第02节 数据访问类&#xff08;DAO&#xff09;第03节 数据Service层第04节 RoomDataBase …

达梦数据库DPI 实现两个数据库数据互通

链接字符串是目标访问链接 目标访问用户名 口令实现 31 里访问33库的数据 如果在31上建立视图访问33的某个表 AS SELECT SZZJ.sys_user.id FROM SZZJ.sys_userszzj31_szzj33;

护眼灯哪些牌子好?五款专业护眼灯品牌排行推荐

普通台灯长时间使用下来&#xff0c;眼睛疲劳、酸涩。但当作业或者工作没有做完的时候&#xff0c;还得硬着头皮撑着。大家是不是经常为这种事情发愁&#xff1f;于是&#xff0c;护眼台灯被设计出来了&#xff0c;但市面上出现的护眼台灯种类多&#xff0c;质量也是难以保证&a…

开发进度网站带后台源码

【源码介绍】 后台地址是&#xff1a;admin.php 后台没有账号密码 这个没有数据库 有能力的可以自己改 【搭建教程】 1.源码上传至虚拟机或者服务器 2.绑定域名和目录 3.访问域名安装&#xff0c; 4.安装完成后就行了 注&#xff1a;资源均网络搬运 仅供测试学习使用&#xff…