游戏实践:扫雷

一.游戏介绍

虽然很多人玩过这个游戏,但还是介绍一下。在下面的格子里,埋的有10颗雷,我们通过鼠标点击的方式,点出你认为不是雷的地方,等到把所有没有雷的格子点完之后,及视为游戏胜利。

上面的数字的意思就是它的一圈的格子有几颗雷。 

 明白了游戏的玩法,那么我们就可以来写一下这个游戏。

二.游戏的实现

我们首先来创建三个文件,分别是用来实现游戏内容的头文件game.c,来测试的text.c文件,还有头文件game.c。

1.游戏的大纲(text.c)

我们可以先把游戏大体的思路给顺下来:

比如在正式写游戏内容的实现之前,我们是不是需要写一下游戏的目录(按1开始游戏,或者按0退出游戏之类的)。

(1)目录(menu函数)

像是游戏的目录:

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

(2)选择开始与结束(text函数)

我想让他按1开始游戏,按0退出游戏,我是不是就可以用一个do  while循环来实现:

void test()
{int input = 0;srand((unsigned int)time(NULL));do{menu();printf("请选择:>");scanf("%d", &input);//1 0 xswitch (input){case 1:game();break;case 0:printf("游戏结束,退出游戏\n");break;default:printf("选择错误,重新选择\n");break;}} while (input);
}

 (3)游戏内部需要做的(game函数)

此时我想按1来进行游戏,我是不是就可以在1的后面加上一个game()函数,来实现游戏内容。我们在game.c文件里实现好这些功能后,在这个函数里直接调用就行了。

注意:这里看不懂没关系,只是想让大家知道这个游戏我们要干啥。

要实现游戏内容,我们就需要初始化棋盘,布置雷,打印棋盘,排查雷。这些都是在game.c文件里来实现,最后在text.c文件里调用就行了。

void game()
{//完成扫雷游戏//mine数组中存放布置好的雷的信息char mine[ROWS][COLS] = { 0 };//数组全部初始化为'0'//show数组中存放排查出的雷的信息char show[ROWS][COLS] = { 0 };//数组全部初始化为'*'//初始化棋盘InitBoard(mine, ROWS, COLS, '0');InitBoard(show, ROWS, COLS, '*');//布置雷//就9*9的棋盘上随机布置10个雷SetMine(mine, ROW, COL);//DisplayBoard(mine, ROW, COL);//打印棋盘DisplayBoard(show, ROW, COL);//排查雷FindMine(mine, show, ROW, COL);
}

2.游戏内容的实现

先把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 InitBoard(char arr[ROWS][COLS], int rows, int cols, char set);//打印棋盘
void DisplayBoard(char arr[ROWS][COLS], int row, int col);//布置雷
void SetMine(char arr[ROWS][COLS], int row, int col);//排查雷
void FindMine(char mine[ROWS][COLS], char show[ROWS][COLS], int row, int col);

(1)初始化棋盘

我们创建一个函数来初始化我们的棋盘:共有四个参数

第一个参数就是我们的棋盘。

第二个参数是我们棋盘的行。

第三个参数是我们棋盘的列。

第四个参数是我们需要把棋盘初始化成什么(比如0,*)。

void InitBoard(char arr[ROWS][COLS], int rows, int cols, char set)
{int i = 0;for (i = 0; i < rows; i++){int j = 0;for (j = 0; j < cols; j++){arr[i][j] = set;//两个for循环实现整个棋盘的初始化}}
}

(2)打印棋盘

这个函数需要三个参数:分别是棋盘,行和列。

void DisplayBoard(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++){int j = 0;printf("%d ", i);for (j = 1; j <= col; j++){printf("%c ", arr[i][j]);}printf("\n");}
}

(3)布置棋盘

这个步骤就很重要了,因为是我们布置雷的过程。

我们想要随机的布置雷的方位,我们就需要利用好time.h。

void SetMine(char arr[ROWS][COLS], int row, int col)
{int count = EASY_COUNT;while (count){int x = rand() % row + 1;//这里的rand函数需要与srand函数配合使用,目的就是生成一个随机数int y = rand() % col + 1;//这里的x和y的范围都是1~10if (arr[x][y] == '0'){arr[x][y] = '1';//注意这里的1和0都是字符count--;}}
}

注意:在使用rand函数生成随机数之前,通常需要调用srand函数来设置随机数种子。随机数种子是一个起始值,它会影响后续rand函数生成的随机数序列。如果每次程序运行时都使用相同的随机数种子,那么每次生成的随机数序列都会相同。

(4)知道雷的数量

因为当我们扫到一个雷的时候,会反馈给我们周围雷的数量,所以要有那么一个步骤:

我们可以把每一个格子都加起来,看看有几个1。

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

如果用上面的方式,会显得有点麻烦,还可以这样写: 

static int GetMineCount(char mine[ROWS][COLS], int x, int y)
{int i = 0;int count = 0;for (i = x - 1; i <= x + 1; i++){int j = 0;for (j = y - 1; j <= y + 1; j++){count += (mine[i][j] - '0');//因为是字符1,所以我们要减去字符0,得到数字1}}return count;
}

(5)排查雷

四个参数:一个是有0和1的参数,一个是让玩家看的参数,另外两个是行和列。

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 (show[x][y] == '*'){if (mine[x][y] == '1'){printf("很遗憾,你被炸死了\n");DisplayBoard(mine, ROW, COL);//被炸死了直接把带有雷的棋盘弄出来break;}else{//该坐标不是雷,就得统计坐标周围有几个雷int count = GetMineCount(mine, x, y);show[x][y] = count + '0';//我们的count是数字,所以要加上字符0的ASCII码值DisplayBoard(show, ROW, COL);win++;}}else{printf("该坐标已经被排查了,重新输入坐标\n");}}else{printf("坐标非法,请重新输入\n");}}if (win == row * col - EASY_COUNT){printf("恭喜你,排雷成功\n");DisplayBoard(mine, ROW, COL);}
}

到这里游戏内容的步骤就完成了。下面的完整的代码分享。

三.完整代码分享

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 InitBoard(char arr[ROWS][COLS], int rows, int cols, char set);//打印棋盘
void DisplayBoard(char arr[ROWS][COLS], int row, int col);//布置雷
void SetMine(char arr[ROWS][COLS], int row, int col);//排查雷
void FindMine(char mine[ROWS][COLS], char show[ROWS][COLS], int row, int col);

text.c文件:

#include "game.h"void menu()
{printf("**********************\n");printf("***** 1. play    *****\n");printf("***** 0. exit    *****\n");printf("**********************\n");
}void game()
{//完成扫雷游戏//mine数组中存放布置好的雷的信息char mine[ROWS][COLS] = { 0 };//数组全部初始化为'0'//show数组中存放排查出的雷的信息char show[ROWS][COLS] = { 0 };//数组全部初始化为'*'//初始化棋盘InitBoard(mine, ROWS, COLS, '0');InitBoard(show, ROWS, COLS, '*');//布置雷//就9*9的棋盘上随机布置10个雷SetMine(mine, ROW, COL);//DisplayBoard(mine, ROW, COL);//打印棋盘DisplayBoard(show, ROW, COL);//排查雷FindMine(mine, show, ROW, COL);
}void test()
{int input = 0;srand((unsigned int)time(NULL));do{menu();printf("请选择:>");scanf("%d", &input);//1 0 xswitch (input){case 1:game();break;case 0:printf("游戏结束,退出游戏\n");break;default:printf("选择错误,重新选择\n");break;}} while (input);
}int main()
{test();return 0;
}

game.c文件


#include "game.h"void InitBoard(char arr[ROWS][COLS], int rows, int cols, char set)
{int i = 0;for (i = 0; i < rows; i++){int j = 0;for (j = 0; j < cols; j++){arr[i][j] = set;}}
}void DisplayBoard(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++){int j = 0;printf("%d ", i);for (j = 1; j <= col; j++){printf("%c ", arr[i][j]);}printf("\n");}
}void SetMine(char arr[ROWS][COLS], int row, int col)
{int count = EASY_COUNT;while (count){int x = rand() % row + 1;int y = rand() % col + 1;if (arr[x][y] == '0'){arr[x][y] = '1';count--;}}
}static int GetMineCount(char mine[ROWS][COLS], int x, int y)
{int i = 0;int count = 0;for (i = x - 1; i <= x + 1; i++){int j = 0;for (j = y - 1; j <= y + 1; j++){count += (mine[i][j] - '0');}}return 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 (show[x][y] == '*'){if (mine[x][y] == '1'){printf("很遗憾,你被炸死了\n");DisplayBoard(mine, ROW, COL);break;}else{//该坐标不是雷,就得统计坐标周围有几个雷int count = GetMineCount(mine, x, y);show[x][y] = count + '0';DisplayBoard(show, ROW, COL);win++;}}else{printf("该坐标已经被排查了,重新输入坐标\n");}}else{printf("坐标非法,请重新输入\n");}}if (win == row * col - EASY_COUNT){printf("恭喜你,排雷成功\n");DisplayBoard(mine, ROW, COL);}
}

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

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

相关文章

python爬虫------- Selenium下篇(二十三天)

&#x1f388;&#x1f388;作者主页&#xff1a; 喔的嘛呀&#x1f388;&#x1f388; &#x1f388;&#x1f388;所属专栏&#xff1a;python爬虫学习&#x1f388;&#x1f388; ✨✨谢谢大家捧场&#xff0c;祝屏幕前的小伙伴们每天都有好运相伴左右&#xff0c;一定要天天…

【爬虫开发】爬虫从0到1全知识md笔记第5篇:Selenium课程概要,selenium的其它使用方法【附代码文档】

爬虫开发从0到1全知识教程完整教程&#xff08;附代码资料&#xff09;主要内容讲述&#xff1a;爬虫课程概要&#xff0c;爬虫基础爬虫概述,,http协议复习。requests模块&#xff0c;requests模块1. requests模块介绍,2. response响应对象,3. requests模块发送请求,4. request…

Python开源工具库使用之词云Wordcloud

文章目录 前言一、基本使用1.1 文本生成词云1.2 配置项 二、进阶用法2.1 自定义形状2.2 自定义着色2.3 自定义词频2.4 中文 三、实际案例3.1 工作报告词云3.2 周杰伦歌词词云 四、总结4.1 优点和局限性4.2 展望未来发展 参考 前言 当我们需要将大量文本数据可视化展示时&#…

【Entity Framework】你要知道EF中功能序列与值转换

【Entity Framework】你要知道EF中功能序列与值转换 文章目录 【Entity Framework】你要知道EF中功能序列与值转换一、序列1.1 基本用法1.2 配置序列设置 二、值转换2.1 配置值转换器2.2 批量配置值转换器2.3 预定义的转换2.4 ValueConverter类2.5 内置转换器 三、应用3.1 简单…

白盒测试详解

&#x1f345; 视频学习&#xff1a;文末有免费的配套视频可观看 &#x1f345; 关注公众号&#xff1a;互联网杂货铺&#xff0c;回复1 &#xff0c;免费获取软件测试全套资料&#xff0c;资料在手&#xff0c;涨薪更快 概念与定义 白盒测试&#xff1a;侧重于系统或部件内部机…

攻防世界1

阅读须知&#xff1a; 探索者安全团队技术文章仅供参考,未经授权请勿利用文章中的技术资料对任何计算机系统进行入侵操作,由于传播、利用本公众号所提供的技术和信息而造成的任何直接或者间接的后果及损失&#xff0c;均由使用者 本人负责&#xff0c;作者不为此承担任何责任,如…

为了执行SQL语句,MySQL的架构是怎样设计的

1. 把MySQL当个黑盒子一样执行SQL语句 上一讲我们已经说到&#xff0c;我们的系统采用数据库连接池的方式去并发访问数据库&#xff0c;然后数据库自己其实也会维护一个连 接池&#xff0c;其中管理了各种系统跟这台数据库服务器建立的所有连接 我们先看下图回顾一下 当我们的…

苹果备忘录误删一段内容怎么恢复?iPhone备忘录恢复的4种方法!收藏!

在使用苹果设备时&#xff0c;备忘录是许多用户常用的工具之一。iPhone备忘录是一款功能强大、易于使用的应用&#xff0c;它能帮助你更好地管理生活和工作。无论你是记录待办事项、设置提醒还是与他人分享信息&#xff0c;备忘录都能满足你的需求。 然而&#xff0c;如果不小…

Mac下载的软件显示文件已损坏,如何解决文件已损坏问题

当在Mac上下载的软件显示文件已损坏时&#xff0c;这可能是因为多种原因导致的&#xff0c;包括网络问题、下载中断、软件未完整下载、文件传输错误等。解决这个问题需要采取一些步骤来排除可能的原因&#xff0c;并尝试修复文件。下面将详细介绍一些常见的解决方法&#xff1a…

单链表链表专题

1 链表的概念 概念&#xff1a;链表是⼀种物理存储结构上⾮连续、⾮顺序的存储结构&#xff0c;数据元素的逻辑顺序是通过链表中的指针链接次序实现的。 链表的结构跟⽕⻋⻋厢相似&#xff0c;淡季时⻋次的⻋厢会相应减少&#xff0c;旺季时⻋次的⻋厢会额外增加⼏节。只 需要…

Laravel 11入门:使用ServBay打造高效开发环境

Laravel 11发布&#xff0c;改进了不少功能。 它引入了更加流畅的应用结构、每秒限速、健康路由等特性。 此外&#xff0c;Laravel还推出了第一方可扩展的WebSocket服务器Laravel Reverb&#xff0c;为你的应用提供强大的实时功能。 在今天的指南中&#xff0c;我将设置一个…

【C语言__动态内存管理__复习篇6】

目录 前言 一、动态内存管理 二、动态内存函数 2.1 malloc 2.2 free 2.3 calloc 2.4 realloc 三、动态内存常见的6个使用错误 3.1 接收malloc/calloc返回的参数后未及时检查是否为NULL 3.2 越界访问动态内存空间 3.3 对非动态开辟的内存使用free释放 3.4 使用free只释放了…