C语言——简易版扫雷

news2024/11/17 13:52:07

目录

前言

​编辑

 游戏规则

 游戏结构的分析

游戏的设计 

使用多文件的好处有以下几点:

 游戏代码实现

框架(test.c)

 game函数(test.c)

InitBoard初始化(game.c)

 Print打印棋盘(game.c)

Setmine设置雷(game.c)

Findmine排查雷(game.c) 

GetMine  

game.h

game.c

exe程序

 Debug和Release的区别



前言

扫雷游戏是一种益智类的游戏,目标是通过揭示方块上的数字来找到不带雷的方块,避免触雷。 


 游戏规则

玩家的目标是根据已翻开的方块信息来推测出地雷的位置,并避开它们。每个方块要么是空白,要么显示数字。数字表示该方块周围八个方向上的地雷数量。通过使用数字信息和逻辑推理,玩家可以确定哪些方块是安全的,哪些是地雷。当玩家翻开所有非地雷方块时,游戏胜利。如果玩家不慎踩到地雷,游戏失败。


 游戏结构的分析

扫雷的过程中,布置的雷和排查出的雷的信息都需要存储,所以我们需要⼀定的数据结构来存储这些 信息。

所以我们首先会想到一个9*9的矩阵。

我们将有雷的设置为“1”,没雷的地方设置为“0”

示例

0123456789
1010000000
2000000000
3000001010
4100100000
5000000000
6000000000
7001000001
8000010000
9100000100

假设我们排查(5,3),在(5,3)周围的一圈有雷,则会在屏幕上记起来。

这就是我们想要设计的样子,但是当我们想访问(6,9)的时候,我访问就会越界,到了界外去了,为了解决这一问题,我们将表格变成11*11的模型,这就能很好的解决这个问题。

0123456789
1010000000
2000000000
3000001010
4100100000
5000000000
6000000000
7001000001
8000010000
9100000100

继续分析,我们已经设置了数字“1”为雷,“0”为非雷,当我们排查到一个雷时,需要将这个雷的信息储存起来,然后将它打印出来,作为排雷的重要参考信息的。。那这个雷的个数信息存放在哪⾥呢?如果存放在布 置雷的数组中,这样雷的信息和雷的个数信息就可能或产⽣混淆和打印上的困难。

将雷和⾮雷的信息不要使⽤数字,使⽤某些字符就⾏,这样就避免冲 突了,但是这样做棋盘上有雷和⾮雷的信息,还有排查出的雷的个数信息,就⽐较混杂,不够⽅便。

 这⾥我们采⽤另外⼀种⽅案,我们专⻔给⼀个棋盘(对应⼀个数组mine)存放布置好的雷的信息,再 给另外⼀个棋盘(对应另外⼀个数组show)存放排查出的雷的信息。这样就互不⼲扰了,把雷布置到 mine数组,在mine数组中排查雷,排查出的数据存放在show数组,并且打印show数组的信息给后期 排查参考。

mine数组

01234567891011
0
1‘1’‘0’‘0’‘0’‘0’‘0’‘0’‘1’‘0’‘0’
2‘0’‘0’‘0’‘0’‘0’‘0’‘0’‘0’‘0’‘1’
3‘0’‘0’‘0’‘0’‘1’‘0’‘0’‘0’‘0’‘0’
4‘0’‘1’‘0’‘0’‘0’‘0’‘0’‘0’‘1’‘0’
5‘0’‘0’‘0’‘0’‘0’‘0’‘0’‘0’‘0’‘0’
6‘0’‘0’‘0’‘0’‘1’‘0’‘0’‘0’‘0’‘0’
7‘1’‘0’‘0’‘0’‘0’‘0’‘0’‘0’‘1’‘0’
8‘0’‘0’‘0’‘0’‘0’‘0’‘0’‘0’‘0’‘0’
9‘0’‘0’‘0’‘0’‘1’‘0’‘0’‘0’‘0’‘0’
10‘0’‘0’‘0’‘0’‘0’‘0’‘0’‘0’‘0’‘0’
11

char mine[11][11] = {0};//⽤来存放布置好的雷的信息
 
char show[11][11] = {0};//⽤来存放排查出的雷的个数信息
 

对于show数组就是将所有‘0’和‘1’换成‘*’这里就不展示了,不水字数了。


游戏的设计 

使用多文件的好处有以下几点:

  • 1. 组织性好:将代码划分到不同的文件中,可以更好地组织和管理代码。每个文件可以分别负责不同的功能模块,使代码结构更清晰。

  • 2. 可维护性强:多文件的编程方式使得修改或更新某个功能模块变得更加简单。只需修改相应的文件,不需要修改整个程序。

  • 3. 可重用性高:将一些常用的函数或功能封装到单独的文件中,可以在多个项目中复用这些代码。

  • 4. 编译效率高:当调用某个函数时,编译器只需要编译包含该函数的文件,而不需要重新编译整个程序,提高了编译速度。

  • 5. 可扩展性强:如果需要添加新的功能模块,只需添加一个新的文件,不会对原有代码造成影响。

  • 6. 可测试性好:每个文件可以独立地进行测试,便于定位和修复问题。

总之,使用多文件的编程方式可以提高代码的组织性、可维护性、可重用性和扩展性,提高编译效率和测试效率。

在这个扫雷中我们需要:

game.h         ⽂件中写游戏需要的数据类型和函数声明等

game.c         ⽂件中写游戏中函数的实现等

text.c            ⽂件中写游戏的测试逻辑


 游戏代码实现

框架(test.c)

我们写个基础的框架。将头文件都放在game.h里面,就需要包含头文件,而我们自己的头文件要用“”。

#define _CRT_SECURE_NO_WARNINGS
#include"game.h"
int main()
{
	text();
	return 0;
}

我们进入text函数是,打印菜单menu(),让玩家进行选择,是进入游戏,还是退出游戏。

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

 函数进来是直接do-while进行打印菜单,然后供玩家选择,选择1进入游戏,0退出游戏,若输入以外的内容则会提示出错误,需要重新输入。

void text()
{
	
	int input = 0;
	srand((unsigned int)time(NULL));//这是播种由函数 rand 使用的随机数发生器。
    //我们先放在这里
	do
	{
		menu();
		printf("请输入你的选择:");
        scanf("%d", &input);
		switch(input)
		{
		case 1:
			game();
			break;
		case 0:
			printf("退出游戏\n");
			break;
		default:
            printf("选择错误,请重新输入:");
			break;
		}
	} while (input);
}

 game函数(test.c)

我们整体思路是将数组初始化,mine初始化为‘0’,show初始化为‘*’;随后布置地雷,我们初始化和布置雷好后都能打印一下看看是否符合我们的要求。最后进行排查地雷,将所有的雷找出来就行了。

void game()
{
	char mine[ROWS][COLS] ;
	char show[ROWS][COLS] ;

	//初始化
	InitBoard(mine, ROWS, COLS, '0'); 
	InitBoard(show, ROWS, COLS, '*');

	//打印棋盘
	Print(show, ROWS, COLS);
	/*Print(mine, ROWS, COLS);*/

	//布置地雷
	Setmine(mine, ROW ,COL);

	//排查地雷
	Findmine(mine,show,ROW,COL);
}

InitBoard初始化(game.c)

对数组mine和show进行初始化,将mine的数组全部初始化为‘0’,show初始化为‘*’。如果只是简单的在数组中初始化化‘0’,‘*’。我们传入应该char set就能够让代码更加灵活,不用在写一样的代码浪费时间和空间。

void InitBoard(char arr[ROWS][COLS], int rols, int cols,char set)
{
	for (int i = 0; i < rols; i++)
	{
		for (int j = 0; j < cols; j++)
		{
			arr[i][j] = set;//'0' , '*'
		}
	}
}

 Print打印棋盘(game.c)

对数组mine和数组show进行打印虽然我们设置的是11*11的棋盘,但是对于玩家来说他们只需要再9*9的棋盘里面进行排查雷就行了,所以我们打印的是9*9的棋盘。在加上坐标,能让玩家更精确的定位。

void Print(char arr[ROWS][COLS], int rols, int cols)
{
	printf("-----扫雷游戏——————\n");//让界面整体更加美观
	for (int i = 0; i < rols - 1 ; i++)
	{
		printf("%d ", i);
	}
	printf("\n");
	for (int i = 1; i < rols - 1; i++)
	{
		printf("%d ", i);
		for (int j = 1; j < cols - 1; j++)
		{
			printf("%c ", arr[i][j]);
		}
		printf("\n");
	}
}

Setmine设置雷(game.c)

我们只需要在9*9的棋盘随机布置雷就行了,这时需要用到rand这个函数rand()%row = 8 再加1。

void Setmine (char board[ROWS][COLS], int row, int col)
{
	int count = minebox;
	while(count)
	{
		//设置行数和列数的随机
		int x = rand() % row + 1;
		int y = rand() % col + 1;
		if (board[x][y] == '0')
		{
			board[x][y] = '1';
			count--;
		}
	}
}

需要的头文件 

 rand---->  #inlcude<stdlib.h>

 time---->  #include<time.h>

Findmine排查雷(game.c) 

开始我们需要踹按创建变量x和y,供玩家输入坐标,我们的坐标只有1~9所以需要  0<x<10,0<y<10作为条件,这个坐标之外的为非法最标,需要重新输入如果玩家排查的坐标是雷,那么游戏结束,并打印棋盘。如果不是雷,我们需要统计一下它周围有几个雷,这就需要统计雷GetMine,将雷的信息传到数组show上,个数统计出来还要加上‘0’;如果仅仅是这样的话游戏是不会结束的,还需要给whle加上条件win< row * col - minebox(minebox是雷的数量),每次没排查到雷级win++。

如果有耐心的坚持去玩,那么肯定是能玩完的,前提是没被炸死哈。

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 - minebox)
	{
		printf("请输入要扫的坐标:");
		scanf("%d %d", &x, &y);
		if (x > 0 && x < row && y >0 && y < col)
		{
			system("cls");
			if (mine[x][y] == '1')
			{
				printf("你被炸死了,游戏失败\n");
				Print(show, ROWS, COLS);
				break;
			}
			else 
			{
				int count = GetMine(mine, x, y);
				show[x][y] = count + '0';
				Print(show, ROWS, COLS);
				win++;
			}
		}
		else
		{
			printf("非法输入\n");
		}
	}
    if (win == row * col - minebox)
	{
		printf("恭喜你,排雷成功\n");
		Print(mine, ROW, COL);
	}

}

system("cls")的头文件跟rand的一样

GetMine  

static是静态的意思,静态函数只能在声明它的文件中可见,其他文件不能引用该函数。

‘1’的值是49;'0'的值是48;‘1’-‘0’=1是个整数。我们将周边的数都加起来-8*‘0’就能得到雷的数。

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

game.h

#define ROW 9 //行数
#define COL 9 //列数

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

#define minebox 10 //雷的个数

#include<stdio.h>   //main函数的头文件
#include<time.h>    //time函数的头文件
#include<stdlib.h>  //system,rand函数的头文件

//初始化棋盘
void InitBoard(char arr[ROWS][COLS], int rols, int cols, char set);


//打印棋盘
void Print(char arr[ROWS][COLS], int rols, int cols);

//设置雷
void Setmine(char arr[ROWS][COLS], int x, int y);

//找雷
void Findmine(char mine[ROWS][COLS], char show[ROWS][COLS], int row, int col);

game.c

同理,game.c要使用game.h的东西也得包含头文件。

到这里,我们这个简易版的扫雷就实现了。

怎么把代码编程可发布的exe程序呢?


exe程序

  1. 将debug模式改为Release模式。
  2. 打开“项目”点击属性。
  3. 打开“C/C++”下的“代码生成”将“运行库”改为多线程(/MT)
  4. “Ctrl+F5”运行程序,这里exe的程序就出现在文件夹里面了。
  5. 打开我们存放代码的文件夹找到x64里面会有两个文件夹,一个是Debug,一个是Relase,点开,我们就能看到一个exe的程序后缀,这样就能直接发给其他人。

 Debug和Release的区别

Debug和release是软件开发中常见的两种构建(build)模式,它们在编译、优化和调试方面有以下区别:

  • 1. 编译方式:Debug模式通常会生成包含调试信息的可执行文件,以便在调试过程中能够准确定位代码的问题。而release模式会采用更高级别的优化和压缩,生成体积更小、性能更高的可执行文件。
  • 2. 优化级别:Debug模式通常会使用较低级别的优化,以便更好地保留源代码的结构和逻辑,使得调试更容易。而release模式会使用更高级别的优化,以提高程序的运行性能。
  • 3. 调试信息:Debug模式会保留更多的调试信息,如变量名、函数名等,以便在调试器中能够查看和修改这些信息。而release模式会舍弃部分调试信息,以减小可执行文件的体积。
  • 4. 异常处理:Debug模式通常会提供更多的异常信息,使得在程序出错时能够更方便地定位问题。而release模式会舍弃部分异常信息,以提高程序的性能和稳定性。

总之,Debug模式适用于开发和调试阶段,能够提供更多的调试信息和更好的可读性;而release模式适用于发布阶段,能够提供更高的性能和较小的体积。在实际开发中,通常会进行Debug模式的开发和调试,然后再切换到release模式进行最终的发布。

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

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

相关文章

【C++初阶】第六站 : 模板初阶

前言&#xff1a; 本章知识点&#xff1a;泛型编程、函数模板、类模板 专栏&#xff1a; C初阶 目录 泛型编程 函数模板 1.函数模板概念 2.函数模板格式 3.函数模板的原理 4.函数模板的实例化 5.模板参数的匹配原则 类模板 类模板的定义格式 类模板的实例化 泛型编程 如何实现一…

Flink实时数仓同步:实时表、流水表、快照表整合实战详解

一、背景 在大数据领域&#xff0c;数据分析、实时数仓已经成为平台上常见的功能之一。无论是进行实时分析还是离线分析&#xff0c;都离不开数仓中的表数据。 特别是在实时分析领域&#xff0c;查阅实时数据、历史数据以及历史变更数据是非常常见的需求。而这些功能的实现主…

基于android的物业管理系统的设计与实现19.8

目录 基于android的物业管理系统的设计与实现 3 摘 要 3 Android property managemengt system 5 Abstract 5 1 绪论 6 1.1 选题背景 6 1.2 课题研究现状 6 1.3 设计研究主要内容 7 1.4 系统主要设计思想 8 2 开发环境 8 2.1 Android系统的结构 8 图2-1 Android系统架构图 9 2…

【C++精简版回顾】21.迭代器,实现迭代器

1.什么是迭代器&#xff1f; 用来遍历容器&#xff0c;访问容器数据。 2.迭代器使用 1.初始化 //初始化 list<int> mylist;//list的整数对象 list<int>::iterator iter;//list内部类&#xff0c;迭代器对象(正向输出) list<int>::reverse_iterator riter;//…

详解数据挖掘

数据挖掘&#xff08;Data Mining&#xff09;&#xff0c;又译为资料探勘、数据采矿&#xff0c;是数据库知识发现&#xff08;Knowledge-Discovery in Databases&#xff0c;简称&#xff1a;KDD&#xff09;中的一个步骤。数据挖掘主要是指从大量的数据中&#xff0c;通过算…

如何选择软文推广渠道?媒介盒子分享

想要做好一个品牌&#xff0c;除了软文文案、推广方式要不断更新外&#xff0c;软文推广渠道也十分重要。有许多企业在创立之初容易踩平台没选好的坑。渠道是品牌触达用户的关键点&#xff0c;今天媒介盒子就来和大家聊聊&#xff1a;如何选择软文发布平台。 一、 了解平台用户…

Java项目:47 ssm007亚盛汽车配件销售业绩管理统+jsp(含文档)

作者主页&#xff1a;源码空间codegym 简介&#xff1a;Java领域优质创作者、Java项目、学习资料、技术互助 文中获取源码 项目介绍 亚盛汽车配件销售业绩管理系统根据调研&#xff0c;确定管理员管理客户&#xff0c;供应商&#xff0c;员工。 管理配件和配件的进货以及出售…

Python中的异常处理及最佳实践【第125篇—异常处理】

Python中的异常处理及最佳实践 异常处理是编写健壮、可靠和易于调试的Python代码中不可或缺的一部分。在本文中&#xff0c;我们将深入探讨Python中的异常处理机制&#xff0c;并分享一些最佳实践和代码示例&#xff0c;以帮助您更好地处理错误情况和提高代码的稳定性。 异常…

VUE3项目学习系列--Axios二次封装(五)

Axios中文文档 | Axios中文网 (axios-http.cn) Axios 是一个基于 promise 网络请求库&#xff0c;作用于node.js 和浏览器中。 它是 isomorphic 的(即同一套代码可以运行在浏览器和node.js中)。在服务端它使用原生 node.js http 模块, 而在客户端 (浏览端) 则使用 XMLHttpRequ…

【SpringCloud微服务实战01】Eureka 注册中心

前言 在 Eureka 架构中,微服务角色有两类: EurekaServer :服务端,注册中心 记录服务信息 心跳监控 EurekaClient :客户端 Provider :服务提供者,例如案例中的 user-service 注册自己的信息到 EurekaS…

java SSM在线学习网站系统myeclipse开发mysql数据库springMVC模式java编程计算机网页设计

一、源码特点 java SSM在线学习网站系统是一套完善的web设计系统&#xff08;系统采用SSM框架进行设计开发&#xff0c;springspringMVCmybatis&#xff09;&#xff0c;对理解JSP java编程开发语言有帮助&#xff0c;系统具有完整的源代码和数据库&#xff0c;系统主要采用…

ubuntu搭建HTTP/3 协议的 Nginx QUIC

ubuntu搭建HTTP/3 协议的 Nginx QUIC 什么是 HTTP/3 和 QUIC&#xff1f; HTTP/3 是一种基于 QUIC (Quick UDP Internet Connections) 协议的 HTTP 协议版本&#xff0c;它是 HTTP/2 的后继者&#xff0c;旨在改进 Web 性能和安全性。 HTTP/3 与之前的 HTTP 协议有很大的不同…

谷粒商城实战(004 整合elasticSearch(es)搜索引擎)

Java项目《谷粒商城》架构师级Java项目实战&#xff0c;对标阿里P6-P7&#xff0c;全网最强 总时长 104:45:00 共408P 此文章包含第125p-第p127的内容 整合es 最好使用Eleasticsearch-Rest-Client 24年改用Java API Client 其实可以直接用js直接调用es 进行查询&#xff0c;这…

力扣大厂热门面试算法题 12-14

12. 整数转罗马数字&#xff0c;13. 罗马数字转整数&#xff0c;14. 最长公共前缀&#xff0c;每题做详细思路梳理&#xff0c;配套Python&Java双语代码&#xff0c; 2024.03.11 可通过leetcode所有测试用例。 目录 12. 整数转罗马数字 解题思路 完整代码 Java Pytho…

5G CA频段组合与带宽的射频标准

先来复习一下我们前面学习过的章节后缀所代表的含义&#xff1a; None Single CarrierA Carrier Aggregation (CA)B Dual-Connectivity (DC)C Supplement Uplink (SUL)D UL MIMOE V2XF Shared spectrum channel accessG Tx Diversity (TxD)I …

【MySQL探索之旅】数据表的基本操作(附带思维导图)

&#x1f4da;博客主页&#xff1a;爱敲代码的小杨. ✨专栏&#xff1a;《Java SE语法》 | 《数据结构与算法》 | 《C生万物》 |《MySQL探索之旅》 ❤️感谢大家点赞&#x1f44d;&#x1f3fb;收藏⭐评论✍&#x1f3fb;&#xff0c;您的三连就是我持续更新的动力❤️ &…

maven运行spring boot项目

我用idea想跑一个整合flowable的spring boot项目&#xff0c;但是跑不起来&#xff0c;原因是jdk版本不够高。但是我的idea是2018版本&#xff0c;最高只能支持到jdk11。就想办法不用idea编译、打包、运行项目。因为spring boot是maven项目&#xff0c;所以可以用maven进行打包…

04-微服务 面试题

1.Spring Cloud 常见的组件有哪些? Spring Cloud 5大组件有哪些? 基础的内容考察回答原则:简单的问题不能答错(一道面试题就能淘汰一个人)新手和老手都要注意面试参考回答: 面试官:Spring Cloud 5大组件有哪些? 候选人:早期我们一般认为的Spring Cloud五大组件是 …

EXCEL根据某列的数字N,增加N-1行相同的数据

因为工作需要&#xff0c;需要将表格数据拆分&#xff0c;类似于相同的订单有6笔&#xff0c;数据表中就是一行数据但是订单数为6&#xff0c;但是需要将其拆分成相同6笔的订单数为1的数据行。 需要使用VBA代码&#xff0c;具体做法如下&#xff1a; Dim i As Long, j As Long…

web3D三维引擎(Direct3D、OpenGL、UE、U3D、threejs)基础扫盲

Hi&#xff0c;我是贝格前端工场的老司机&#xff0c;本文介绍文web3D的几个引擎&#xff0c;做个基础扫盲&#xff0c;如果还不能解决问题&#xff0c;可以私信我&#xff0c;搞私人订制呦。 三维引擎是指用于创建和渲染三维图形的软件框架。它们通常提供了图形处理、物理模拟…