首页 > 其他分享 >C语言_扫雷

C语言_扫雷

时间:2024-09-02 19:53:33浏览次数:17  
标签:ROWS int COLS C语言 char 扫雷 board printf

引言:当我们C语言学习完数组的时候,可以尝试一些小的游戏来提升自己,比如下面的这个扫雷。

玩法介绍:我们需要在棋盘中选中地雷以外的安全区域,直到最后剩下全部的地雷为止,即可获胜。

技能要求:分支语句以及循环语句、函数、数组。

接下来我将创建三个文件:具备函数声明、宏定义、所需库函数的头文件以及还有两个函数实现以及测试游戏逻辑的源文件。

文件名如下:

  • game.h(头文件)
  • game.c(函数实现的源文件)
  • test.c(游戏逻辑的源文件)

项目大纲:

写项目之前,我们要明白自己想写什么,在脑海里要有一个清晰的流程。

  • 游戏逻辑
  • 菜单
  • 创建并初始化棋盘
  • 打印棋盘(方便观察)
  • 布置地雷
  • 排查地雷
  • 判断状态(继续排查、排查完毕、踩到地雷)

对于扫雷项目的一些事项

对于扫雷我们会创建两个棋盘,第一种对于我们可以直观看见棋盘上的地雷以及安全区域的分布情况,还有一种就是无法查看的棋盘。

我们布置地雷也自然会把地雷布置在可视化的棋盘中。

当我们进行排查地雷的时候,会生成对周围区域的一个检索,也就是说可视化的棋盘会对周围的区域进行一个遍历,然后返回地雷的个数放在不可视化的棋盘中的指定坐标上。

另外我们需要知道,当地雷排查到一定程度就可以获得胜利,这个也是我们下面需要注意的地方。

game.h (头文件)

//游戏相关的函数声明、宏定义以及调用库函数

#include <stdio.h>
#include <time.h>
#include <stdlib.h>

#define ROW 9
#define COL 9

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

#define landmine 80

//初始化棋盘
void Initboard(char board[ROWS][COLS], int rows, int cols, char p);
//打印棋盘
void revealboard(char board[ROWS][COLS], int rows, int cols);
//布置雷
void SetMine(char board[ROWS][COLS], int rows, int cols);
//排查雷
void FindMine(char board[ROWS][COLS], char board_show[ROWS][COLS], int rows, int cols);

game.c (函数实现的源文件)

初始化棋盘

void Initboard(char board[ROWS][COLS], int rows, int cols, char p)
{
	int i = 0;
	for (i = 0; i < rows; i++)
	{
		int j = 0;
		for (j = 0; j < cols; j++)
		{
			board[i][j] = p;
		}
	}
}

打印棋盘

void revealboard(char board[ROWS][COLS], int rows, int cols)
{
	int i = 0;
	printf("------------扫雷游戏------------\n");
	for (i = 0; i < rows - 1; i++)
	{
		if (i == 0)
		{
			printf("\\ ");
			continue;
		}
		printf("%d ", i);
	}
	printf("\n");
	for (i = 1; i < rows - 1; i++)
	{
		int j = 0;
		printf("%d ", i);
		for (j = 1; j < cols - 1; j++)
		{
			printf("%c ", board[i][j]);
		}
		printf("\n");
	}
	printf("------------扫雷游戏------------\n");
}

可视化棋盘以及不可视化棋盘预览:

在这里插入图片描述

在这里插入图片描述

布置地雷

void SetMine(char board[ROWS][COLS], int rows, int cols)
{
	int i = 0;
	int x = 0;
	int y = 0;
	while(1)
	{
		x = rand() % rows;
		y = rand() % rows;
		//落棋区间 x:1-9 y:1-9
		//筛选出去不在区间内的坐标
		if (((x < 1) || (x > 9)) || ((y < 1) || (y > 9)))
		{
			continue;
		}
		//判断有无重复
		if (board[x][y] == '0')
		{
			board[x][y] = '*';
			i++;
			if (i == landmine)
			{
				break;
			}
		}
	}
}

布置地雷的可视化棋盘预览

在这里插入图片描述

返回指定坐标周围地雷的个数

int GetMine(char board[ROWS][COLS], int x, int y)
{
	int i = 0;
	int count = 0;
	//范围也就是在:x-1,y-1 到 x+1,y+1
	for (i = x - 1; i <= x + 1; i++)
	{
		int j = 0;
		for (j = y - 1; j <= y + 1; j++)
		{
			if (i == x && j == y)
			{
				continue;
			}
			else if(board[i][j] == '*')
			{
				count++;
			}
			else
			{
				;
			}
		}
	}
	return count;
}

返回地雷个数并且在显示在不可视化的棋盘中:

在这里插入图片描述

排查雷并统计雷的个数,确保排完地雷结束游戏

void FindMine(char board[ROWS][COLS], char board_show[ROWS][COLS], int rows, int cols)
{
	int x = 0;
	int y = 0;
	int count = 0;
	//设置死循环不近可以为输入错误的坐标提供重新输入的几回
	//而且也可以重复进行排查
	while (1)
	{
		printf("请输入需要排查的坐标:");
		scanf("%d %d", &x, &y);
		//当我们输入指定范围坐标以外的坐标时,就不会往下面执行
		if (x >= 1 && x <= 9 && y >= 1 && y <= 9)
		{
			if (board[x][y] == '*')
			{
				printf("很遗憾,你被炸死了!!!\n");
				break;
			}
			//判断周围有几个地雷
			else
			{
				//返回雷的个数
				int ret = GetMine(board, x, y);
				//在隐藏棋盘中的指定下标输入地雷个数
				board_show[x][y] = '0' + ret;
				count++;
				//打印看看效果
				revealboard(board_show, ROWS, COLS);
				//判断有没有将地雷全部排查掉
				if (count == ((ROW * COL) - landmine))
				{
					printf("恭喜你地雷已经全部排完了!!!\n");
					break;
				}
			}
		}
		else
		{
			printf("输入有误,请重新检查!!!\n");
		}
	}
}

排雷成功预览:
在这里插入图片描述

test.c (游戏逻辑的源文件)

#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 };
	//初始化棋盘
	Initboard(mine, ROWS, COLS, '0');
	Initboard(show, ROWS, COLS, '*');
	//打印棋盘
	/*revealboard(mine, ROWS, COLS);
	revealboard(show, ROWS, COLS);*/
	//布置雷
	SetMine(mine, ROWS, COLS);
	revealboard(mine, ROWS, COLS);
	//排查雷
	FindMine(mine, show, ROWS, COLS);
}

//游戏界面选择
int main()
{
	srand((unsigned int)time(NULL)); 
	int input = 0;
	do
	{
		menu();
		printf("请选择:");
		scanf("%d", &input);
		switch (input)
		{
		case 1:
			game();
			break;
		case 0:
			printf("欢迎下次再来!!!\n");
			break;
		default:
			printf("输入有误,请重新输入!!!\n");
			break;
		}
	} while (input);
	return 0;
}

以上就是关于扫雷的所有代码以及注释了,如果对你有帮助的话,不妨留下一个赞吧~

标签:ROWS,int,COLS,C语言,char,扫雷,board,printf
From: https://blog.csdn.net/m0_74445256/article/details/141728888

相关文章

  • c语言编译器IDE的6键钢琴程序代码
    #include<stdio.h>#include<SDL2/SDL.h>#include<SDL2/SDL_mixer.h>//FunctionforloadingmusictoMix_MusicstaticMix_Music*loadMusic(constchar*path){Mix_Music*music=Mix_LoadMUS(path);if(music==NULL){fprintf(stderr,“M......
  • c语言编译器IDE小钢琴程序代码
    #include<stdio.h>#include<SDL2/SDL.h>#include<SDL2/SDL_mixer.h>//FunctionforloadingmusictoMix_MusicstaticMix_Music*loadMusic(constchar*path){Mix_Music*music=Mix_LoadMUS(path);if(music==NULL){fprintf(stderr,“M......
  • C语言指针(进阶内容)
    目录1.指针的基本概念:2.不同类型的指针的意义是什么?3.野指针是什么?4.指针加减整数 5.指针运算(指针-指针)6.一级指针和二级指针以及多级指针是什么意思?7.指针数组8.数组指针9.函数指针1.指针的基本概念:1.指针就是个变量,用来存放地址,地址唯一标识一块内存空......
  • 【树莓派开发】使用树莓派在Linux环境下编写C语言代码
    文章目录前言1.创建test.c文件2.编译运行该文件3.编译并链接两个源文件结语前言如何使用树莓派编译C语言代码呢?21年暑假的时候,学习编程的劲头高涨,然后冲动消费买了个树莓派4B……结果压根不会用,吃灰了半年不过现在已经学完了C语言,也接触了一丢丢Linux系统下的gcc指令,可以尝试用它......
  • 【C语言】文件操作(详解)
    文章目录1.为什么需要文件?2.什么是文件?2.1文件分类2.2文件名3.文件的使用3.1文件指针3.2打开和关闭文件3.2.1文件使用方式3.2.2标准输入输出流3.3文件输入输出函数3.3.1字符输入输出实现文件拷贝3.3.2文本行输入输出3.3.3格式化输入输出3.3.4二进制输入输出3.3.5sscanf/sprintf函......
  • 【C语言】顺序表详解,灵活运用所学知识
    文章目录前言1.什么是顺序表?1.1线性表2.编写你的顺序表!2.0赛前准备2.1初始化2.2容量检查2.3打印顺序表2.4尾插和尾删2.5头插和头删2.6插入和删除2.7查找和更改3.菜单一些err总结前言顺序表是我们学习数据结构第一阶段的必经之路什么是顺序表,且听我慢慢道来本篇博客用到的......
  • 【C语言】数据结构-栈(顺序表实现)
    文章目录前言1.什么是栈2.栈的实现3.敲代码!3.1头文件3.2函数实现4.知识巩固,来道OJ!结语前言在之前的数据结构学习中,我们学习了顺序表、链表这两种结构顺序表:博客链接1单链表:博客链接2链表OJ:博客链接3除了单链表以外,还有一个结构,是双向带头循环链表。这个链表的形式如下头节点的......
  • 【方法分享】准大学生如何自学C语言?
    准大学生如何自学C语言文章目录准大学生如何自学C语言一、找寻一个优质教学视频二、使用优质编译器三、安排一个电子笔记需不需要一个ipad来辅助学习?四、除工具以外的学习方法1.专心致志,不要让其他东西分心2.多练习,多练习,多练习3.复习!结语虽说现在已经接近我们最长的且无忧无虑的......
  • C语言函数递归(含扫雷进阶思路)
    文章目录一、什么是递归二、递归的使用思路和限制条件1.递归的使用思路2.递归的限制条件三、递归的举例举例1:求n的阶乘2.举例2:顺序打印⼀个整数的每⼀位四、递归与迭代对比五、递归与迭代对比举例七、扫雷进阶思路一、什么是递归  递归是学习C语⾔函数绕不开的......
  • 基于C语言的选择排序算法
    一、选择排序算法的基本原理        选择排序算法是一种简单直观的排序算法。其基本原理为:        首先,将待排序的数组划分为已排序和未排序两部分。初始时,已排序部分为空,未排序部分为整个数组。        在每一轮排序中,从未排序部分找出最小(或最大)......