C语言项目实战——扫雷

目录

1.前言

2.完整流程

2.1规划书

2.2代码部分

2.2.1文件的结构设计

2.2.2变量的创建

2.2.3菜单的基本实现

2.2.4初始化期棋盘

2.2.5输出完整棋盘

2.2.6埋雷的实现

2.2.7查询周围雷的数量

2.2.8扫雷的实现

 2.2.9完整代码

3.总结


1.前言

哈喽大家好吖,今天笔者就前期基本所学手搓了一个扫雷的小项目,考察的知识点很简单仅有基本的分支,循环,函数调用,数组的基本应用,相信你也可以做到滴,快来试试喔~

2.完整流程

2.1规划书

虽然完成扫雷的代码量并不大,但我们还是要理清楚我们需要完成的功能和运行的基本逻辑。

 

这是一张扫雷的基本界面,通过观察和实际游玩可分析得出:

基本功能包括:

  • 游戏可以通过菜单实现继续玩或者退出游戏
  • 扫雷的棋盘是9*9的格⼦
  • 默认随机布置10个雷
  • 可以排查雷
  • 如果位置不是雷,就显⽰周围有⼏个雷
  • 如果位置是雷,就炸死游戏结束
  • 把除10个雷之外的所有非雷都找出来,排雷成功,游戏结束

在梳理完需要实现的功能以后,就可以开始逐步实现以上功能咯。(当然笔者目前无法做出上述照片如此精良的扫雷,只能做一个简略版滴~)

2.2代码部分

2.2.1文件的结构设计

扫雷的代码量虽然不大,但是已经不适合用单文件来完成了,于是乎我们就把,游戏逻辑的基本实现,相关游戏函数的声明,以及游戏相关函数的实现分别打包,方便管理。

我们创建以下三个文件:

test.c //游戏的测试逻辑 
game.c //游戏中函数的实现
game.h //游戏需要的数据类型和函数声明等

完成后是这个样子。

另外,在正式敲代码前,我们应先在game.h这个头文件下加上

#pragma once

 #pragma once 是一个预处理指令,用来防止头文件的内容在一个编译单元中被多次包含。这与传统的头文件保护(也称为包含保护或头文件守卫)非常相似,后者通常使用#ifndef, #define, 和 #endif来实现。

2.2.2变量的创建

在扫雷这个游戏中,我们一共需要用到以下几个变量:

  • 棋盘一(origin)用于记录是否埋雷
  • 棋盘二(infor)用于记录雷的相关信息
  • 常量row,col分别用于记录扫雷中最核心的9*9区域
  • 常量rows,cols用于记录大棋盘11*11区域
  • 常量easy用于记录所要标记的雷

 实现分别如下(常量记得要在head.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 10//记录简单难度有十个雷

 俩个棋盘的创建我放在了game函数中。

void game() {char origin[rows][cols];char infor[rows][cols];//创建俩个棋盘//初始化棋盘init(origin, rows, cols, '0');init(infor, rows, cols, '*');//打印棋盘用于调试print(infor, row, col);//开始埋雷setboom(origin, row, col);//print(origin, row, col);//开始排雷findboom(origin, infor, row, col);}

2.2.3菜单的基本实现

为了菜单的基本美观,我们在实现的时候可以进行反复调试,代码如下:

void menu() {printf("------------------------\n");printf("------欢迎来到扫雷------\n");printf("------------------------\n");printf("----------菜单----------\n");printf("-------1.开始游戏-------\n");printf("-------2.结束游戏-------\n");//基础菜单实现printf("---------3.彩蛋---------\n");return;
}
int main() {int n = 0;//创建选项变量srand((unsigned int)time(NULL));do {menu();printf("请选择选项\n");scanf("%d", &n);switch (n) {case 1:game(); break;case 2:printf("已退出\n"); n = 0; break;case 3:printf("彩蛋\n"); n = 0; break;default:printf("请重新输入\n");}} while (n);//do while的巧妙利用return 0;
}

菜单的基本功能的实现是通过switch语句, 如果要玩游戏输入1进入game函数,如果选择退出或者彩蛋选项,将会在case语句中将n置为0从而跳出do while语句,最终实现是这个样子:

2.2.4初始化期棋盘

从这一部分往下,就是开始要分别写出实现每个小功能的函数,记得在game.c中写完函数要在game.h完成声明。

对于初始化,我们将origin这个棋盘全部置为字符'0',将infor这个棋盘全部置为'*'。

game.c中:

void init(char board[rows][cols], int rows1, int cols1, char set) {int i = 0;for (i = 0; i < rows; i++){int j = 0;for (j = 0; j < cols; j++){board[i][j] = set;}}
}

函数的声明我在这就不完成了,就是一个小小体力活, 最后完整代码会体现出来。效果如下:

2.2.5输出完整棋盘

输出的时候,除了将9*9的核心区域打印出来,还要记得标记行列(1~9)以及x,y轴,增加可玩性。

void print(char board[rows][cols], int row1, int col1) {printf("-----蘸豆---爽-----\n");printf("0 1 2 3 4 5 6 7 8 9 y\n");for (int i = 1; i <= row1; i++) {printf("%d ", i);for (int j = 1; j <= col1; j++) {printf("%c ", board[i][j]);}printf("\n");}printf("x\n");
}

最终效果如下:

2.2.6埋雷的实现

由于埋雷的随机性,这里需要一个c语言中的随机数函数rand。

rand用法:

在C语言中,rand() 是一个标准库函数,用于生成一个伪随机数。这个函数定义在 <stdlib.h> 头文件中。rand() 函数返回一个在 0 到 RAND_MAX 之间的整数,其中 RAND_MAX 是在 <stdlib.h> 中定义的一个宏,其值至少为 32767

由于 rand() 生成的是伪随机数,因此在每次程序运行时,如果不进行任何设置,它将生成相同的随机数序列。为了得到不同的随机数序列,通常在程序开始时使用 srand() 函数来设置随机数生成器的种子。种子的值通常来自于系统时间或者其他变化的数据。

这里就是用时间作为种子:srand((unsigned int)time(NULL)); (在主函数中书写了出来)

埋雷的时候我们使用while循环进行操作,另外有一个小小的注意点,如果该点已经被埋过雷了,注意排除,代码如下:

void setboom(char board[rows][cols], int row1, int col1) {int count = easy;//记录雷数while (count) {int x = rand() % 9 +1;int y = rand() % 9 +1;if (board[x][y] == '0') {board[x][y] = '1';count--;}}
}

如果该位置为'0'就进行埋雷操作,如果为'1'就重复该循环。

2.2.7查询周围雷的数量

在正式开始扫雷函数的实现前,我们还需要先实现查找以该点为中心(非雷)周围八个格子雷的数量。实现这个功能有很多方式,这我采用了ascii码的一些基本特性。 

int set_infor(char board[rows][cols], int x, int y) {return (board[x - 1][y - 1] + board[x][y - 1] +board[x - 1][y] + board[x][y + 1]+ board[x + 1][y - 1] + board[x + 1][y]+ board[x + 1][y + 1] + board[x - 1][y + 1] - 8 * '0');
}

因为我创建的俩个棋盘都是char类型,所以说这个方法就可以使用。

2.2.8扫雷的实现

这块部分比较核心,以下是思路讲解:

  • 创建下x,y变量用于记录坐标,创建win用于标记当前排除了多少个区域。
  • 主体函数为while循环,循环条件为win变量仍然小于9*9总格数减去总雷数,那么胜利条件自然是win变量等于9*9总格数减去总雷数。
  • 接下来讲解while主体里面思路:
  • 先让玩家输入x,y坐标,并判断该坐标是否为坐标上的合法目标,如果不是就让玩家重新输入,如果该坐标合法就进行下一步。
  • 判断如果玩家选择的坐标为雷,则该玩家被炸死并跳出循环。
  • 如果选择的非雷,就打印出该位置上周围八个格子的雷数。

最终代码如下:

void findboom(char origin1[rows][cols], char infor1[rows][cols], int row1, int col1) {int x = 0;int y = 0;int win = 0;//用于记录当前排掉了多少个雷while (win < row1 * col1 - easy) {printf("请输入你要排查的x坐标>__");scanf("%d", &x);printf("请输入你要排查的y坐标>__");scanf("%d", &y);if (x > 0 && y > 0 && x <= row1 && y <= col1) {if (origin1[x][y] == '1') {printf("你个憨批你被炸死了\n");break;}else {int num = set_infor(origin1, x, y);infor1[x][y] = num + '0';print(infor1, col1, row1);win++;}}else {printf("该坐标不在棋盘上,你这个笨蛋\n请重新输入\n");}}if (win == row1 * col1 - easy) {printf("恭喜你!你胜利咯\n");print(origin1, row1, col1);}
}

 运行如下图:

 2.2.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 10//记录简单难度有十个雷void menu();void init(char board[rows][cols], int rows1, int cols1, char set);void setboom(char board[rows][cols], int row1, int col1);void print(char board[rows][cols], int row1, int col1);int set_infor(char board[rows][cols], int x, int y);void findboom(char origin1[rows],char infor1[rows][cols],int row1,int col1);

 game.c:

#define  _CRT_SECURE_NO_WARNINGS 1
//相关函数的定义
#include"game.h"
void menu() {printf("------------------------\n");printf("------欢迎来到扫雷------\n");printf("------------------------\n");printf("----------菜单----------\n");printf("-------1.开始游戏-------\n");printf("-------2.结束游戏-------\n");//基础菜单实现printf("---------3.彩蛋---------\n");return;
}void init(char board[rows][cols], int rows1, int cols1, char set) {int i = 0;for (i = 0; i < rows; i++){int j = 0;for (j = 0; j < cols; j++){board[i][j] = set;}}
}void print(char board[rows][cols], int row1, int col1) {printf("-----蘸豆---爽-----\n");printf("0 1 2 3 4 5 6 7 8 9 y\n");for (int i = 1; i <= row1; i++) {printf("%d ", i);for (int j = 1; j <= col1; j++) {printf("%c ", board[i][j]);}printf("\n");}printf("x\n");
}void setboom(char board[rows][cols], int row1, int col1) {int count = easy;//记录雷数while (count) {int x = rand() % 9 +1;int y = rand() % 9 +1;if (board[x][y] == '0') {board[x][y] = '1';count--;}}
}int set_infor(char board[rows][cols], int x, int y) {return (board[x - 1][y - 1] + board[x][y - 1] +board[x - 1][y] + board[x][y + 1]+ board[x + 1][y - 1] + board[x + 1][y]+ board[x + 1][y + 1] + board[x - 1][y + 1] - 8 * '0');
}void findboom(char origin1[rows][cols], char infor1[rows][cols], int row1, int col1) {int x = 0;int y = 0;int win = 0;//用于记录当前排掉了多少个雷while (win < row1 * col1 - easy) {printf("请输入你要排查的x坐标>__");scanf("%d", &x);printf("请输入你要排查的y坐标>__");scanf("%d", &y);if (x > 0 && y > 0 && x <= row1 && y <= col1) {if (origin1[x][y] == '1') {printf("你个憨批你被炸死了\n");break;}else {int num = set_infor(origin1, x, y);infor1[x][y] = num + '0';print(infor1, col1, row1);win++;}}else {printf("该坐标不在棋盘上,你这个笨蛋\n请重新输入\n");}}if (win == row1 * col1 - easy) {printf("恭喜你!你胜利咯\n");print(origin1, row1, col1);}
}

test.c:

#define  _CRT_SECURE_NO_WARNINGS 1//游戏逻辑的实现
#include"game.h"void game() {char origin[rows][cols];char infor[rows][cols];//创建俩个棋盘//初始化棋盘init(origin, rows, cols, '0');init(infor, rows, cols, '*');//打印棋盘用于调试print(infor, row, col);//开始埋雷setboom(origin, row, col);//print(origin, row, col);//开始排雷findboom(origin, infor, row, col);}int main() {int n = 0;//创建选项变量srand((unsigned int)time(NULL));do {menu();printf("请选择选项\n");scanf("%d", &n);switch (n) {case 1:game(); break;case 2:printf("已退出\n"); n = 0; break;case 3:printf("彩蛋是这个~\n"); n = 0; break;default:printf("请重新输入\n");}} while (n);//do while的巧妙利用return 0;
}

3.总结

以上就是扫雷的基本功能实现了,其实还有许多可以扩展,例如点开一个非雷区自动展开周围其他非雷区域,调整难度等,大家都可以自己尝试尝试哦~。都看到这里了,希望大家对我多多支持喔,大家一起进步~

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

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

相关文章

【C++打怪之路Lv3】-- 类和对象(上)

&#x1f308; 个人主页&#xff1a;白子寰 &#x1f525; 分类专栏&#xff1a;C打怪之路&#xff0c;python从入门到精通&#xff0c;数据结构&#xff0c;C语言&#xff0c;C语言题集&#x1f448; 希望得到您的订阅和支持~ &#x1f4a1; 坚持创作博文(平均质量分82)&#…

开发环境搭建:Windows 桌面应用程序

文章目录 前言1、开发环境准备2、Hello World !3、发布总结 前言 操作系统&#xff1a;Windows 10 企业版 LTSC 1809 IDE&#xff1a;Microsoft Visual Studio 2022 Community 说明&#xff1a;Windows 10 企业版 LTSC 1809 不支持 .NET 4.8.1 详情请查看官方说明文档 1、开发…

The Log-Structured Merge-Tree (LSM-Tree) 论文阅读笔记

原论文&#xff1a;The Log-Structured Merge-Tree (LSM-Tree) LSM-Tree的简介和关键技术要点 LSM-Tree&#xff08;Log-Structured Merge-Tree&#xff09;是一种为高吞吐量读写操作优化的数据结构&#xff0c;特别适用于写入密集型的应用场景。它由Patrick O’Neil等人开发…

基于streamlit快速部署机器学习项目(Public URL)

基于streamlit的AIGC项目前端展示 1.Streamlit 简介与入门1.1 安装 Streamlit1.2 开发Streamlit应用程序1.3 启动并运行1.3.1 本地运行1.3.2 部署 现在LLM技术发展迅速&#xff0c;很多人在学习的时候&#xff0c;都想展示效果&#xff0c;并且想部署在服务器上&#xff0c;但是…

【人工智能基础】线性回归实验分析

实验使用到的库&#xff1a;numpy、matplotlib、scikit-learn 实验使用的开发环境&#xff1a;anaconda、jupyter 一、线性回归 线性回归就是使用一个线性函数&#xff08;多项式回归可以是曲线&#xff09;去拟合给定的训练集&#xff0c;测试时&#xff0c;对输入的x值&#…

Jammy@Jetson Orin - Tensorflow Keras Get Started: Concept

JammyJetson Orin - Tensorflow & Keras Get Started: Concept 1. 源由2. 模型2.1 推理流程2.1.1 获取图像2.1.2 算法识别2.1.3 判断决策 2.2 理想情况2.2.1 多因素输入2.2.2 理想识别概率 2.3 学习过程2.3.1 标记训练集2.3.2 损失函数2.3.3 训练网络2.3.4 渐进方法 3. 总…

jvm(JVM快速入门、stack栈、堆、GC垃圾回收、Arthas)

文章目录 1. JVM快速入门1.1. 结构图1.2. 类加载器ClassLoader1.3. 执行引擎Execution Engine1.4. 本地接口Native Interface1.5. Native Method Stack1.6. PC寄存器(程序计数器)1.7. Method Area方法区 2. stack栈3. 堆3.1. 堆体系概述3.1.1. 新生区3.1.2. 老年代3.1.3. 永久代…

python基础知识点(蓝桥杯python科目个人复习计划66)

今日复习内容&#xff1a;算法双周赛 第一题&#xff1a;疯狂星期六 题目描述&#xff1a; 麦肯鸡是一家名声在外的汉堡店&#xff0c;他们最近推出了一份名为vivo50的套餐&#xff0c;只需要在门口大声喊出vivo50&#xff0c;就可以获得这个套餐。 现在&#xff0c;请你打…

了解ASK模块STX883Pro和超外接收模块SRX883Pro的独特之处 STX883Pro模块具有以下特点:

高发射功率&#xff1a;STX883Pro具有较高的发射功率&#xff0c;可实现长距离的信号传输&#xff0c;适用于需要覆盖广泛区域的应用场景。 高频率稳定性&#xff1a;具备稳定的频率输出&#xff0c;确保信号传输的可靠性和一致性&#xff0c;避免频率漂移导致的通信故障。 大…

C++ | Leetcode C++题解之第48题旋转图像

题目&#xff1a; 题解&#xff1a; class Solution { public:void rotate(vector<vector<int>>& matrix) {int n matrix.size();// 水平翻转for (int i 0; i < n / 2; i) {for (int j 0; j < n; j) {swap(matrix[i][j], matrix[n - i - 1][j]);}}//…

Thread方法具体解析

对于run方法 如果该线程是使用单独的 Runnable run 对象构造的&#xff0c;则调用该 Runnable 对象的 run 方法&#xff1b;否则&#xff0c;此方法不执行任何操作并返回。 对于start方法 导致该线程开始执行&#xff1b; Java虚拟机调用该线程的run方法。 这里介绍一个快捷键…

Windows Vscode ModuleNotFoundError: No module named

故障现象&#xff1a; Windows Vscode 经常会遇到模块路径查找失败的异常。 如运行2_from_import_test.py后&#xff0c;报错&#xff1a; 发生异常: ModuleNotFoundError No module named programmer File "D:\leolab\programmer\2_from_import_test.py", line 8…

什么是数字化运营?

目录 一、什么是数字化运营&#xff1f; 二、数字化运营的重要性是什么&#xff1f; 三、数字化运营的具体步骤和措施是什么&#xff1f; 四、数据化决策是什么&#xff1f; 一、什么是数字化运营&#xff1f; 数字化运营是利用数字技术和数据分析来优化企业的业务流程和运…

C语言 | Leetcode C语言题解之第51题N皇后

题目&#xff1a; 题解&#xff1a; int solutionsSize;char** generateBoard(int* queens, int n) {char** board (char**)malloc(sizeof(char*) * n);for (int i 0; i < n; i) {board[i] (char*)malloc(sizeof(char) * (n 1));for (int j 0; j < n; j) board[i][…

【unity】三维数学应用(计算线和面的交点)

【unity】三维数学应用&#xff08;计算线和面的交点&#xff09; 实现方法有多种&#xff0c;下面介绍一种简单的方法。利用一个点指向面上任意点的向量&#xff0c;到该面法线的投影长度相同的基本原理&#xff0c;结合相似三角形既可以求出交点。 原理 如下图 GD组成的线段…

c++图论基础(1)

目录 无向图 无向图度 无向图性质 有向图 有向图度 有向图性质 图的分类&#xff1a; 稀疏图&#xff1a; 稠密图&#xff1a; 零图&#xff1a; 有向完全图&#xff1a; 无向完全图&#xff1a; 度序列&#xff1a; 图是由顶点集合(简称点集)和顶点间的边(简称边…

Weblogic JMS

简介 全称:WebLogic Server的Java Messaging Service(JMS) WebLogic JMS 是与 WebLogic Server 平台紧密集成的企业级消息传递系统。 Java Message Service (JMS) API 是一种消息传递标准,允许基于 Java Platform Enterprise Edition (Java EE) 的应用程序组件创建、发送、…

windows rabbitMq安装

一、Erlang 环境准备 下载安装包 跟我们跑java项目&#xff0c;要装jdk类似。rabbitMQ是基于Erlang开发的&#xff0c;因此安装rabbitMQ服务器之前&#xff0c;需要先安装Erlang环境。 官网直接下载windows直装版本&#xff1a;https://www.erlang.org/downloads 无脑安装&a…

【RocketMQ知识点总结-1】

文章目录 RocketMQ介绍RocketMQ架构&#xff1a;NameServer:BrokerProducerTopic&#xff08;主题&#xff09;&#xff1a;Queue&#xff08;队列&#xff09;&#xff1a;Message&#xff08;消息&#xff09;&#xff1a; RocketMQ的工作流程RocketMQ的使用场景异步消息传递…

预训练扩散模型用于即插即用的医学图像增强

文章目录 Pre-trained Diffusion Models for Plug-and-Play Medical Image Enhancement摘要本文方法Image Enhancement with Denoising AlgorithmPre-Trained Diffusion Models for Plug-and-play Medical Image Enhancement 实验结果 Pre-trained Diffusion Models for Plug-a…