C语言实现扫雷游戏完整版

2024-05-11 07:20

本文主要是介绍C语言实现扫雷游戏完整版,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

游戏介绍:

目录

游戏介绍:

游戏框架:

游戏具体功能实现:

棋盘的定义:

棋盘初始化:

棋盘打印:

棋盘布置雷:

棋盘扫雷:

爆炸展开一片:

获取周围八个格子的雷数:

游戏源码:

test.c:

game.h:

game.c:


扫雷游戏相信大家都玩过,大概就是点击一个格子,如果这个格子是雷就被炸死,不是雷,就会显示周围八个格子有几个雷,如果玩家把没有雷的格子点完了就赢了。

游戏框架:

我们还是分为三个文件来写:test.c:用来写游戏整体的逻辑。game.c:写代码中最关键的部分,游戏基本功能实现的函数。game.h:写需要的头文件,常量,函数定义等。

游戏具体功能实现:

棋盘的定义:

首先第一步我们就需要知道雷盘(棋盘)是一个个格子的,所以我们就可以用二维数组来实现。我们看到扫雷游戏,有展现给玩家的雷盘,上面我们看不到哪里有雷,但是如果只有一个二维数组就无法实现,所以我们定义两个二维数组,一个展现给玩家,一个内部操作:

我们的ROWS和COLS这样子定义其一是因为方便修改,其二是因为我们展现出的棋盘要比内部棋盘少两行和两列。

为什么要这样:因为我们想到如果点击雷盘的最边上的一格,在数组上,我们得扫描周围八个格子,会越界操作,如果我们不越界,那就得加上很多限制条件,条件太多可能会把自己弄晕,所以有的大佬想出给我们的数组再加上两行和两列,这样我们依旧扫描周围八个但是就不会越界了,展现给玩家的只有内部的格子。

棋盘初始化:

因为我们有两个棋盘得放不一样的元素,所以初始化函数:

set变量就可以实现我们想让数组初始化什么就初始化什么。

所以内部棋盘都初始化为0,展现棋盘都初始化为 * 。

棋盘打印:

在这里我们就用ROW和COL来遍历棋盘,从下标1开始。

棋盘布置雷:

布置雷需要注意的:

1.只在内部格子布置

2.格子里必须是0才放,以防少放了

我们放雷的时候只需要在需要展现的棋盘上放就行了,所以 i 和 j 这两个随机值要在1——9的范围里,之所以我们用1来表示雷,后面我会解释。

棋盘扫雷:

逻辑是让玩家输入坐标,如果踩到雷了就退出告诉玩家你失败了。如果没有踩到雷,就扫描周围八个格子看有多少个雷,然后显示到格子上,如果坐标格子周围八个格子没有雷就展开周围八个格子,直到把没有雷的格子点完就算赢。

爆炸展开一片:

像这样点击一个之后爆炸式展开该怎么实现,我们是扫描坐标周围八个格子都没有雷的话就把他设成空格,如果看周围八个格子的其中一个周围也没有雷再把他设成空格,这样子可以形成函数递归,我们来实现这个递归:

获取周围八个格子的雷数:

这个时候我们把没有雷设成0,雷设成1 的简单办法就来了,如果没有这样设置,我们就要多设置判断语句。如果是 0 和 1,我们知道数组里是字符0 和 1,根据ASCLL表发现字符的1 减去 0 就是数字1了,所以我们可以把周围格子加起来再一起减去字符0 就是最终结果。

游戏源码:

test.c:

#include"game.h"void meun()
{printf("************************\n");printf("*******  1.  play ******\n");printf("*******  0.  exit ******\n");printf("************************\n");
}void game()
{char mine[ROWS][COLS];char show[ROWS][COLS];InitBoard(mine, ROWS, COLS, '0');InitBoard(show, ROWS, COLS, '*');//打印棋盘DisplayBoard(show, ROW, COL);//布置雷SetMine(mine, ROW, COL);//DisplayBoard(mine, ROW, COL);//排雷FindMine(mine, show, ROW, COL);}int main()
{srand((unsigned int)time(NULL));int input = 0;;do{meun();printf("请选择:");scanf("%d", &input);switch (input){case 1:game();break;case 0:printf("退出游戏\n");break;default:printf("输入错误,重新输入\n");break;}} while (input);
}

game.h:

#include<stdio.h>
#include<stdlib.h>
#include<time.h>
#include<string.h>
#include<Windows.h>#define ROW 9
#define COL 9#define ROWS ROW + 2
#define COLS COL + 2#define Easy_play 20//雷的个数//初始化棋盘
void InitBoard(char board[ROWS][COLS], int rows, int cols, char set);//打印棋盘
void DisplayBoard(char board[ROWS][COLS], int row, int col);//布置雷
void SetMine(char mine[ROWS][COLS], int row, int col);//排查雷
void FindMine(char mine[ROWS][COLS], char show[ROWS][COLS], int row, int col);

game.c:

#include"game.h"//初始化棋盘
void InitBoard(char board[ROWS][COLS], int rows, int cols, char set)
{for (int i = 0; i < rows; i++){for (int j = 0; j < cols; j++){board[i][j] = set;}}
}//打印棋盘
void DisplayBoard(char board[ROWS][COLS], int row, int col)
{printf("----------扫雷-------------\n");for (int i = 1; i <= row; i++){for (int j = 1; j <= col; j++){printf(" %c ", board[i][j]);}printf("\n");}printf("----------扫雷-------------\n");
}void SetMine(char mine[ROWS][COLS], int row, int col)
{int m = Easy_play;while (m){int i = rand() % row + 1;//1 --- 9int j = rand() % col + 1;//要棋盘里必须是0 的时候再放if (mine[i][j] == '0'){mine[i][j] = '1';m--;}}
}int GetMineCount(char mine[ROWS][COLS], int x, int y)
{//因为我们的棋盘里放的‘0’和‘1’所以,我们周围八个加起来就行,但是因为他们是字符,所以只要 - ‘0’,就会变成数字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 My_FindMine(char mine[ROWS][COLS], char show[ROWS][COLS], int row, int col, int x, int y, int* pw)
{if (x >= 1 && x <= row && y >= 1 && y <= col)//要在排查范围内{int c = GetMineCount(mine, x, y);if (c == 0){(*pw)++;show[x][y] = ' ';//这个就可以把坐标变成空格,还可以防止死递归for (int i = x - 1; i <= x + 1; i++){for (int j = y - 1; j <= y + 1; j++){if (show[i][j] == '*')//限制条件,防止查过的再次查到,造成死递归。{My_FindMine(mine, show, row, col, i, j, pw);}}}}else{show[x][y] = c + '0';(*pw)++;}}
}void FindMine(char mine[ROWS][COLS], char show[ROWS][COLS], int row, int col)
{int x = 0;int y = 0;int win = 0;int* pw = &win;while (win < row * col - Easy_play){printf("请输入你的坐标:->");scanf("%d %d", &x, &y);if (x > 0 && x <= row && y > 0 && y <= col){if (mine[x][y] == '1'){system("cls");printf("你被炸死了,真遗憾\n");DisplayBoard(mine, ROW, COL);break;}else{if (show[x][y] != '*'){printf("这个坐标输入过了,请重新输入\n");continue;}else{My_FindMine(mine, show, row, col, x, y, pw);//爆炸张开一片system("cls");//清空屏幕DisplayBoard(show, ROW, COL);}}}else{printf("坐标错了,请重新输入\n");}}if (win == row * col - Easy_play){printf("恭喜你,游戏被你结束了\n");}
}

这篇关于C语言实现扫雷游戏完整版的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/978874

相关文章

C++中unordered_set哈希集合的实现

《C++中unordered_set哈希集合的实现》std::unordered_set是C++标准库中的无序关联容器,基于哈希表实现,具有元素唯一性和无序性特点,本文就来详细的介绍一下unorder... 目录一、概述二、头文件与命名空间三、常用方法与示例1. 构造与析构2. 迭代器与遍历3. 容量相关4

C++中悬垂引用(Dangling Reference) 的实现

《C++中悬垂引用(DanglingReference)的实现》C++中的悬垂引用指引用绑定的对象被销毁后引用仍存在的情况,会导致访问无效内存,下面就来详细的介绍一下产生的原因以及如何避免,感兴趣... 目录悬垂引用的产生原因1. 引用绑定到局部变量,变量超出作用域后销毁2. 引用绑定到动态分配的对象,对象

SpringBoot基于注解实现数据库字段回填的完整方案

《SpringBoot基于注解实现数据库字段回填的完整方案》这篇文章主要为大家详细介绍了SpringBoot如何基于注解实现数据库字段回填的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以了解... 目录数据库表pom.XMLRelationFieldRelationFieldMapping基础的一些代

Java HashMap的底层实现原理深度解析

《JavaHashMap的底层实现原理深度解析》HashMap基于数组+链表+红黑树结构,通过哈希算法和扩容机制优化性能,负载因子与树化阈值平衡效率,是Java开发必备的高效数据结构,本文给大家介绍... 目录一、概述:HashMap的宏观结构二、核心数据结构解析1. 数组(桶数组)2. 链表节点(Node

Java AOP面向切面编程的概念和实现方式

《JavaAOP面向切面编程的概念和实现方式》AOP是面向切面编程,通过动态代理将横切关注点(如日志、事务)与核心业务逻辑分离,提升代码复用性和可维护性,本文给大家介绍JavaAOP面向切面编程的概... 目录一、AOP 是什么?二、AOP 的核心概念与实现方式核心概念实现方式三、Spring AOP 的关

从基础到高级详解Go语言中错误处理的实践指南

《从基础到高级详解Go语言中错误处理的实践指南》Go语言采用了一种独特而明确的错误处理哲学,与其他主流编程语言形成鲜明对比,本文将为大家详细介绍Go语言中错误处理详细方法,希望对大家有所帮助... 目录1 Go 错误处理哲学与核心机制1.1 错误接口设计1.2 错误与异常的区别2 错误创建与检查2.1 基础

一文详解Python如何开发游戏

《一文详解Python如何开发游戏》Python是一种非常流行的编程语言,也可以用来开发游戏模组,:本文主要介绍Python如何开发游戏的相关资料,文中通过代码介绍的非常详细,需要的朋友可以参考下... 目录一、python简介二、Python 开发 2D 游戏的优劣势优势缺点三、Python 开发 3D

Python实现字典转字符串的五种方法

《Python实现字典转字符串的五种方法》本文介绍了在Python中如何将字典数据结构转换为字符串格式的多种方法,首先可以通过内置的str()函数进行简单转换;其次利用ison.dumps()函数能够... 目录1、使用json模块的dumps方法:2、使用str方法:3、使用循环和字符串拼接:4、使用字符

Linux下利用select实现串口数据读取过程

《Linux下利用select实现串口数据读取过程》文章介绍Linux中使用select、poll或epoll实现串口数据读取,通过I/O多路复用机制在数据到达时触发读取,避免持续轮询,示例代码展示设... 目录示例代码(使用select实现)代码解释总结在 linux 系统里,我们可以借助 select、

Linux挂载linux/Windows共享目录实现方式

《Linux挂载linux/Windows共享目录实现方式》:本文主要介绍Linux挂载linux/Windows共享目录实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录文件共享协议linux环境作为服务端(NFS)在服务器端安装 NFS创建要共享的目录修改 NFS 配