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

扫雷

    • 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.rhkb.cn/news/268458.html

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

相关文章

PHP【swoole】

前言 Swoole官方文档&#xff1a;Swoole 文档 Swoole 使 PHP 开发人员可以编写高性能高并发的 TCP、UDP、Unix Socket、HTTP、 WebSocket 等服务&#xff0c;让 PHP 不再局限于 Web 领域。Swoole4 协程的成熟将 PHP 带入了前所未有的时期&#xff0c; 为性能的提升提供了独一无…

springboot197基于springboot的毕业设计系统的开发

简介 【毕设源码推荐 javaweb 项目】基于springbootvue 的毕业设计系统的开发 适用于计算机类毕业设计&#xff0c;课程设计参考与学习用途。仅供学习参考&#xff0c; 不得用于商业或者非法用途&#xff0c;否则&#xff0c;一切后果请用户自负。 看运行截图看 第五章 第四章 …

中小企业的人才测评,给HR的招聘解决方案

中小企业、初创企业在人才招聘上&#xff0c;通常都只能依靠领导的慧眼识人。鉴于招聘人员的数量少&#xff0c;队伍非常精简&#xff0c;那么这个方式也是不错的&#xff0c;老板用慧眼观察&#xff0c;尤其是适合找到跟老板脾性相投的人&#xff0c;共同创业是个不错的选择方…

[回归指标]R2、PCC(Pearson’s r )

R2相关系数 R2相关系数很熟悉了&#xff0c;就不具体解释了。 皮尔逊相关系数&#xff08;PCC&#xff09; 皮尔逊相关系数是研究变量之间线性相关程度的量&#xff0c;R方和PCC是不同的指标。R方衡量x和y的接近程度&#xff0c;PCC衡量的是x和y的变化趋势是否相同。R方是不…

那些壁纸,不只是背景

1、方小童在线工具集 网址&#xff1a; 方小童 该网站是一款在线工具集合的网站&#xff0c;目前包含PDF文件在线转换、随机生成美女图片、精美壁纸、电子书搜索等功能&#xff0c;喜欢的可以赶紧去试试&#xff01;

JavaWeb之 创建 Web项目,使用Tomcat 部署项目,使用 Maven 构建Web项目(一万八千字详解)

目录 前言3.1 Tomcat 简介3.1.1 什么是 Web服务器3.1.2 Tomcat 是什么3.1.3 小结 3.2 Tomcat 的基本使用3.2.1 下载 Tomcat3.2.2 安装 Tomcat3.2.3 卸载 Tomcat3.2.4 启动 Tomcat3.2.5 关闭 Tomcat3.2.6 配置 Tomcat3.2.7 在 Tomcat 中部署 Web项目 3.3 在 IDEA 中创建 Web 项目…

算法43:动态规划专练(最长回文子串 力扣5题)---范围模型

之前写过一篇最长回文子序列的博客算法27&#xff1a;最长回文子序列长度&#xff08;力扣516题&#xff09;——样本模型 范围模型-CSDN博客 在那一篇博客中&#xff0c;回文是可以删除某些字符串组成的。比如&#xff1a; 字符串为&#xff1a;a1b3c4fdcdba&#xff0c; 那…

汽车大灯尾灯的车灯罩破损破裂裂纹等问题用什么胶可以修复??

汽车大灯尾灯破裂可以使用硅酮玻璃胶或者环氧树脂胶进行修复。 环氧树脂胶的优点主要包括&#xff1a; 粘接力强&#xff1a;环氧树脂胶也具有很高的粘接力&#xff0c;可以有效地将裂缝两侧的材料粘合在一起&#xff0c;确保牢固和持久的修复效果。内聚强度大&#xff1a;环…

为啥要用C艹不用C?

在很多时候&#xff0c;有人会有这样的疑问 ——为什么要用C&#xff1f;C相对于C优势是什么&#xff1f; 最近两年一直在做Linux应用&#xff0c;能明显的感受到C带来到帮助以及快感 之前&#xff0c;我在文章里面提到环形队列 C语言&#xff0c;环形队列 环形队列到底是怎么回…

自学高效备考2025年AMC8数学竞赛:2000-2024年AMC8真题解析

今天继续来随机看五道AMC8的真题和解析&#xff0c;根据实践经验&#xff0c;对于想了解或者加AMC8美国数学竞赛的孩子来说&#xff0c;吃透AMC8历年真题是备考最科学、最有效的方法之一。下面的五道题目如果你能在8分钟内做对&#xff08;主要结果对&#xff0c;无需过程&…

一些C语言知识

C语言的内置类型&#xff1a; char short int long float double C99中引入了bool类型&#xff0c;用来表示真假的变量类型&#xff0c;包含true&#xff0c;false。 这个代码的执行结果是什么&#xff1f;好好想想哦&#xff0c;坑挺多的。 #include <stdio.h>int mai…

观成科技:加密C2框架Covenant流量分析

工具介绍 Covenant是一个基于.NET的开源C2服务器&#xff0c;可以通过HTTP/HTTPS 控制Covenant agent&#xff0c;从而实现对目标的远程控制。Covenant agent在与C2通信时&#xff0c;使用base64/AES加密载荷的HTTP隧道构建加密通道。亦可选择使用SSL/TLS标准加密协议&#xf…

【InternLM 实战营笔记】基于 InternLM 和 LangChain 搭建你的知识库

准备环境 bash /root/share/install_conda_env_internlm_base.sh InternLM升级PIP # 升级pip python -m pip install --upgrade pippip install modelscope1.9.5 pip install transformers4.35.2 pip install streamlit1.24.0 pip install sentencepiece0.1.99 pip install a…

【推荐算法系列十七】:GBDT+LR 排序算法

排序算法经典中的经典 参考 推荐系统之GBDTLR 极客时间 手把手带你搭建推荐系统 课程 逻辑回归&#xff08;LR&#xff09;模型 逻辑回归&#xff08;LR,Logistic Regression&#xff09;是一种传统机器学习分类模型&#xff0c;也是一种比较重要的非线性回归模型&#xff…

js监听网页iframe里面元素变化其实就是监听iframe变化

想要监听网页里面iframe标签内容变化&#xff0c;需要通过监听网页dom元素变化&#xff0c;然后通过查询得到iframe标签&#xff0c;再通过iframe.contentWindow.document得到ifram内的document&#xff0c;然后再使用选择器得到body元素&#xff0c;有了body元素&#xff0c;就…

2024年 前端JavaScript Web APIs 第一天 笔记

1.1 -声明变量const优先 1.2 -DOM树和DOM对象 1.3 -获取DOIM元素 1.4 -DOM修改元素内容以及年会抽奖 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8" /><meta http-equiv"X-UA-Compatible" content&quo…

【JAVA重要知识 | 第三篇】深入理解并暴打AQS原理、ReentrantLock锁

文章目录 3.深入理解AQS、ReentrantLock3.1AQS3.1.1AQS简介3.1.2核心结构&#xff08;1&#xff09;设计模型&#xff08;2&#xff09;组成部分&#xff08;3&#xff09;State关键字 3.1.3实现的两类队列&#xff08;1&#xff09;同步队列①CLH②Node③主要行为 img条件队列…

Maven实战(2)之搭建maven私服

一, 背景: 如果使用国外镜像,下载速度比较慢; 如果使用阿里云镜像,速度还算OK,但是假如网速不好的时候,其实也是比较慢的; 如果没有网的情况下更加下载不了. 二, 本地仓库、个人/公司私服、远程仓库关系如下: 三, 下载安装nexus私服 略

【代码】Android|获取压力传感器、屏幕压感数据(大气压、原生和Processing)

首先需要分清自己需要的是大气压还是触摸压力&#xff0c;如果是大气压那么就是TYPE_PRESSURE&#xff0c;可以参考https://source.android.google.cn/docs/core/interaction/sensors/sensor-types?hlzh-cn。如果是触摸压力就是另一回事&#xff0c;我需要的是触摸压力。 不过…

在golang中使用protoc

【Golang】proto生成go的相关文件 推荐个人主页&#xff1a;席万里的个人空间 文章目录 【Golang】proto生成go的相关文件1、查看proto的版本号2、安装protoc-gen-go和protoc-gen-go-grpc3、生成protobuff以及grpc的文件 1、查看proto的版本号 protoc --version2、安装protoc-…