C语言扫雷游戏完整实现(上)

news2024/11/20 21:55:50

文章目录

  • 前言
  • 一、新建好头文件和源文件
  • 二、实现游戏菜单选择功能
  • 三、定义游戏函数
  • 四、初始化棋盘
  • 五、 打印棋盘函数
  • 六、布置雷函数
  • 七、玩家排雷菜单
  • 八、标记功能的菜单
  • 九、标记功能菜单的实现
  • 总结


前言

C语言从新建文件到游戏菜单,游戏函数,初始化棋盘,打印棋盘,布置雷函数,玩家排雷菜单,标记功能菜单及实现等的操作。

下半部分: C语言扫雷游戏完整实现(下)

一、新建好头文件和源文件

  • 库函数头文件常数都定义在game.h头文件中
  • 在test.c源文件和game.c源文件中通过#include "game.h"引入game.h头文件即可
    1. 在test.c源文件中实现菜单以及游戏函数的调用等功能。
    1. 在game.h头文件中主要完成游戏函数的声明等功能。
    1. 在game.c源文件中完成函数的定义/实现工程。

在这里插入图片描述

二、实现游戏菜单选择功能

// game.h 头文件
#include <stdio.h>

// test.c 源文件
#include "game.h"


// 定义游戏菜单函数
void menu()
{
	printf("**********************\n");
	printf("*****   1. Play  *****\n");
	printf("*****   0. Quit  *****\n");
	printf("**********************\n");
}
int main()
{
	int input = 0; // 定义输入菜单选项的变量
	do
	{
		// 游戏菜单
		menu();
		printf("请选择:>");
		scanf("%d", &input);
		// 不同选择对应不同结果
		switch (input)
		{
		case 1:
			printf("扫雷游戏开始!!!\n");
			break;
		case 0:
			printf("推出游戏!!!\n");
			break;
		default:
			printf("选择错误,请重新输入\n");
			break; // 因为要重新输入所以从菜单开始循环
		}
	} while (input); // 输入为0退出游戏,同时停止循环
	return 0;
}

三、定义游戏函数

  • 以9×9为例
  • 需要首先定义 11 × 11(为了便于检测棋盘边边位置的雷) 的棋盘(二维数组)
  • 打印9×9的棋盘即可
  • -需要定义两个棋盘
  • 一个用来存放布置雷的信息
  • 一个用来存放玩家操作信息
// game.h 头文件
#include <stdio.h>

#define ROW 9
#define COL 9

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

// test.c 源文件
#include "game.h"
// 定义游戏菜单函数
void menu()
{
	printf("**********************\n");
	printf("*****   1. Play  *****\n");
	printf("*****   0. Quit  *****\n");
	printf("**********************\n");
}
--------------------------------------------------------------
void game()
{
	// 定义两个二维数组
	char mine[ROWS][COLS] = { 0 };
	char show[ROWS][COLS] = { 0 };

	// 初始化棋盘
}
---------------------------------------------------------------
int main()
{
	int input = 0; // 定义输入菜单选项的变量
	do
	{
		// 游戏菜单
		menu();
		printf("请选择:>");
		scanf("%d", &input);
		// 不同选择对应不同结果
		switch (input)
		{
		case 1:
			printf("扫雷游戏开始!!!\n");
	-------------------------------------------------
			game();
	-------------------------------------------------
			break;
		case 0:
			printf("推出游戏!!!\n");
			break;
		default:
			printf("选择错误,请重新输入\n");
			break; // 因为要重新输入所以从菜单开始循环
		}
	} while (input); // 输入为0退出游戏,同时停止循环
	return 0;
}

四、初始化棋盘

  • 将存放布置雷信息的棋盘初始化为 ‘0’。----设计 ‘0’表示没有雷,‘1’表示雷。
  • 将玩家操作信息的棋盘初始化为 ‘*’。
// test.c --game函数
void game()
{
	// 定义两个二维数组
	char mine[ROWS][COLS] = { 0 };
	char show[ROWS][COLS] = { 0 };

	// 初始化棋盘
	InitBoard(mine, ROWS, COLS, '0');
	InitBoard(show, ROWS, COLS, '*');
}

// game.h 头文件
#include <stdio.h>

#define ROW 9
#define COL 9

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

// 初始化棋盘函数的声明
void InitBoard(char board[ROWS][COLS], int rows, int cols, char sign);

// game.c 源文件
#include "game.h"

// 初始化棋盘函数的定义
void InitBoard(char board[ROWS][COLS], int rows, int cols, char sign)
{
	int i = 0;
	for (i = 0; i < rows; i++)
	{
		int j = 0;
		for (j = 0; j < cols; j++)
		{
			board[i][j] = sign;
		}
	}
}

五、 打印棋盘函数

  • 只打印9×9的棋盘,所以传入参数 ROW 和 COL。
  • 打印棋盘的同时还可以打印出每行每列的数字,方便玩家操作。
  • 同时打印出分割线例如 -----------扫雷游戏---------------
//test.c 源文件
void game()
{
	// 定义两个二维数组
	char mine[ROWS][COLS] = { 0 };
	char show[ROWS][COLS] = { 0 };

	// 初始化棋盘
	InitBoard(mine, ROWS, COLS, '0');
	InitBoard(show, ROWS, COLS, '*');

	// 打印棋盘
	DisplayBoard(mine, ROW, COL);
	DisplayBoard(show, ROW, COL);
}

//game.h 头文件
#include <stdio.h>

#define ROW 9
#define COL 9

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

// 初始化函数声明
void InitBoard(char board[ROWS][COLS], int rows, int cols, char sign);


// 打印函数声明
void DisplayBoard(char board[ROWS][COLS], int row, int col);

//game.c 源文件
// 打印棋盘函数定义
void DisplayBoard(char board[ROWS][COLS], int row, int col)
{
	// 打印游戏开头分割线
	printf("-----扫雷游戏------\n");
	int i = 0;
	// 打印每一列数字
	for (i = 0; i <= row; 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 ", board[i][j]);
		}
		printf("\n");
	}
	// 打印游戏结尾分割线
	printf("-----扫雷游戏------\n");
}

目前可以实现的效果如下图
在这里插入图片描述

六、布置雷函数

  • 在初始化棋盘和打印棋盘之间,应该先布置雷。在mine棋盘(数组)中布置。
  • 在game.h文件中定义常量 EASY_COUNT 来表示雷的数量。
  • 雷只用在9×9的范围内布置即可,所以传入参数 ROW 和 COL。
  • 通过生成9×9范围内的随机数,来布置雷。
  • 用到rand函数,引入 stdlib.h 和 time.h 头文件。
  • 为了保证随机数的随机性,需要在test.c的主函数一开始使用srand函数如下图
    在这里插入图片描述
//test.c 源文件 -- game函数
void game()
{
	// 定义两个二维数组
	char mine[ROWS][COLS] = { 0 };
	char show[ROWS][COLS] = { 0 };

	// 初始化棋盘
	InitBoard(mine, ROWS, COLS, '0');
	InitBoard(show, ROWS, COLS, '*');

	// 布置雷
	SetBoard(mine, ROW, COL);

	// 打印棋盘
	DisplayBoard(mine, ROW, COL);
	DisplayBoard(show, ROW, COL);
}

//game.h 头文件
#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 board[ROWS][COLS], int rows, int cols, char sign);


// 打印函数声明
void DisplayBoard(char board[ROWS][COLS], int row, int col);

// 布置雷函数声明
void SetBoard(char board[ROWS][COLS], int row, int col);

//game.c 源文件
// 布置雷函数定义
void SetBoard(char board[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 (board[x][y] == '0')
		{
			board[x][y] = '1';
			count--;
		}
	}
}

七、玩家排雷菜单

  • 玩家排雷之前要有标记,排雷,以及退出终止的菜单。
  • 菜单选择需要玩家输入,所以定义变量,如若出入错误,循环输入。
//test.c 源文件 --- game函数
void game()
{
	int choose = 0;
	// 定义两个二维数组
	char mine[ROWS][COLS] = { 0 };
	char show[ROWS][COLS] = { 0 };

	// 初始化棋盘
	InitBoard(mine, ROWS, COLS, '0');
	InitBoard(show, ROWS, COLS, '*');

	// 布置雷
	SetBoard(mine, ROW, COL);

	// 打印棋盘
	DisplayBoard(mine, ROW, COL);
	DisplayBoard(show, ROW, COL);

	// 玩家开始排雷
	do
	{
		printf("****************\n");
		printf("*** 1. 标记  ***\n");
		printf("*** 2. 排雷  ***\n");
		printf("*** 0. 退出  ***\n");
		printf("****************\n");
		printf("请选择:>");
		scanf("%d", &choose);

		if (1 == choose)
		{
			printf("标记功能\n");
		}
		else if (2 == choose)
		{
			printf("排雷功能\n");
		}
		else if (0 == choose)
		{
			printf("退出游戏\n");
			break;
		}
		else
		{
			printf("输入错误,请重新输入\n");
		}
	} while (choose);

}

实现效果如下图:
在这里插入图片描述

八、标记功能的菜单

  • 标记功能是玩家操作信息,传入show棋盘(数组)。
  • 玩家只在9×9棋盘中标记,所以使用 ROW 和 COL。
  • 标记功能应该有 1. 标记位置 2. 取消标记 3. 不标记了等三个功能。
  • 标记功能实现的同时对标记的雷进行计数,方便判断胜利。
//test.c 源文件 -- game函数
void game()
{
	int choose = 0;
	// 定义两个二维数组
	char mine[ROWS][COLS] = { 0 };
	char show[ROWS][COLS] = { 0 };

	// 初始化棋盘
	InitBoard(mine, ROWS, COLS, '0');
	InitBoard(show, ROWS, COLS, '*');

	// 布置雷
	SetBoard(mine, ROW, COL);

	// 打印棋盘
	DisplayBoard(mine, ROW, COL);
	DisplayBoard(show, ROW, COL);

	// 玩家开始排雷
	do
	{
		printf("****************\n");
		printf("*** 1. 标记  ***\n");
		printf("*** 2. 排雷  ***\n");
		printf("*** 0. 退出  ***\n");
		printf("****************\n");
		printf("请选择:>");
		scanf("%d", &choose);

		if (1 == choose)
		{
			printf("标记功能\n");
			SignBoard(show, mine, ROW, COL);
		}
		else if (2 == choose)
		{
			printf("排雷功能\n");
		}
		else if (0 == choose)
		{
			printf("退出游戏\n");
			break;
		}
		else
		{
			printf("输入错误,请重新输入\n");
		}
	} while (choose);

}

//game.h 头文件
#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 board[ROWS][COLS], int rows, int cols, char sign);


// 打印函数声明
void DisplayBoard(char board[ROWS][COLS], int row, int col);

// 布置雷函数声明
void SetBoard(char board[ROWS][COLS], int row, int col);

// 标记功能函数声明
int SignBoard(char show[ROWS][COLS],char mine[ROWS][COLS], int row, int col);

//game.c 源文件 ---SignBoard函数
// 标记功能函数定义
int SignBoard(char show[ROWS][COLS], char mine[ROWS][COLS], int row, int col)
{
	int choose = 0;
	do
	{
		printf("******************\n");
		printf("*** 1.标记位置 ***\n");
		printf("*** 2.取消标记 ***\n");
		printf("*** 0.不标记了 ***\n");
		printf("******************\n");
		printf("请选择:>");
		scanf("%d", &choose);

		if (1 == choose)
		{
			printf("标记位置\n");
		}
		else if (2 == choose)
		{
			printf("取消标记\n");
		}
		else if (0 == choose)
		{
			printf("不标记了\n");
			break;
		}
		else
		{
			printf("输入错误,请重新输入\n");
		}

	} while (choose);
}

实现效果如下图:
在这里插入图片描述

九、标记功能菜单的实现

  • 标记位置需要玩家输入
  • 需要判断合法性
  • 需要判断是否被排查
  • 定义两个计数变量,若两变量相等,并且等于雷的个数时,判定排雷成功
  • 每次标记完成或取消标记后,打印棋盘
//game.c 源文件---SignBoard函数完整
// 标记功能函数定义
int SignBoard(char show[ROWS][COLS], char mine[ROWS][COLS], int row, int col)
{
	int x = 0;
	int y = 0;
	int choose = 0;
	static int sum = 0; // 定义标记的总数
	static int count = 0; // 定义标记的雷的总数
	do
	{
		printf("******************\n");
		printf("*** 1.标记位置 ***\n");
		printf("*** 2.取消标记 ***\n");
		printf("*** 0.不标记了 ***\n");
		printf("******************\n");
		printf("请选择:>");
		scanf("%d", &choose);

		if (1 == choose)
		{
			printf("请输入标记坐标(空格隔开):>");
			scanf("%d %d", &x, &y);

			if (x >= 1 && x <= row && y >= 1 && y <= col) // 判断坐标合法性
			{
				if (show[x][y] == '*') // 判断坐标是否排查过
				{
					show[x][y] = '$';
					sum++; // 没标记一个加1
					if (mine[x][y] == '1')
					{
						count++; // 标记的位置如果是雷 加1
					}
				}
				else
				{
					printf("坐标已经被排查过了,请勿重复排查\n");
				}
			}
			else
			{
				printf("超出棋盘范围,请重新输入\n");
			}
			DisplayBoard(show, ROW, COL);
		}
		else if (2 == choose)
		{
			printf("请输入标记坐标(空格隔开):>");
			scanf("%d %d", &x, &y);

			if (x >= 1 && x <= row && y >= 1 && y <= col) // 判断坐标合法性
			{
				if (show[x][y] == '$') // 判断坐标是否标记过
				{
					show[x][y] = '*';
					sum--; // 没标记一个加1
					if (mine[x][y] == '1')
					{
						count--; // 标记的位置如果是雷 加1
					}
				}
				else
				{
					printf("坐标未被标记,请重新选择\n");
				}
			}
			else
			{
				printf("超出棋盘范围,请重新输入\n");
			}
			DisplayBoard(show, ROW, COL);
		}
		else if (0 == choose)
		{
			// 不标记直接跳转
			break;
		}
		else
		{
			printf("输入错误,请重新输入\n");
		}
	if (sum == count && count == EASY_COUNT)
	{
		sum = 0;
		count = 0;
		return EASY_COUNT;
	}
	else
	{
		return 0;
	}

	} while (choose);
}
  • 把雷的个数设置为1进行测试
  • 主要过程是
    1. 先标记一个不是雷的位置。
    1. 再标记雷的位置。
    1. 最后取消标记不是雷的位置。
    1. 测试是否判定成功。
  • 效果如下图:
    在这里插入图片描述

总结

C语言从新建文件到游戏菜单,游戏函数,初始化棋盘,打印棋盘,布置雷函数,玩家排雷菜单,标记功能菜单及实现等的操作及实现效果。

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

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

相关文章

网工交换基础——生成树协议(01)

一、生成树的技术概述 1、技术背景 二层交换机网络的冗余性导致出现二层环路&#xff1a; 人为因素导致的二层环路问题&#xff1a; 二层环路带来的网络问题&#xff1a; 生成树协议的概念&#xff1a; STP(Spanning Tree Protocol)是生成树协议的英文缩写。该协议可应用于在网…

如何分析和优化慢sql语句

前言 sql查询速度比较慢容易成为性能瓶颈,这时我们可以优化我们的sql语句或数据库表 一般sql语句执行很慢的种类分为: 1.聚合查询 2.多表查询 3.表数据量过大查询 4.深度分页查询 这四种的前三种都可以通过优化sql语句来优化sql查询速度 正文 聚合查询 我们可以通过尝…

机器人视觉教学实训平台

一&#xff1a;功能概述 1.1、功能简介 机器人视觉教学实训平台基于睿尔曼机器人与海康机器视觉产品&#xff0c;面向机器人视觉系统应用而开发设计&#xff0c;产品涵盖机器人系统、工业视觉系统、自动化控制系统、计算机编程系统&#xff0c;可以在一台设备上进行多种与机器…

C++初阶学习第三弹——类与对象(上)——初始类与对象

前言&#xff1a; 在前面&#xff0c;我们已经初步学习了C的一些基本语法&#xff0c;比如内敛函数、函数重载、缺省参数、引用等等&#xff0c;接下来我们就将正式步入C的神圣殿堂&#xff0c;首先&#xff0c;先给你找个对象 目录 一、类与对象是什么&#xff1f; 二、类的各…

Git 工作原理

Git 工作原理 | CoderMast编程桅杆https://www.codermast.com/dev-tools/git/git-workspace-index-repo.html Workspace&#xff1a;工作区Index / Stage&#xff1a;暂存区Repository&#xff1a;仓库区&#xff08;或本地仓库&#xff09;Remote&#xff1a;远程仓库 Git 一…

如何优雅的实现 iframe 多层级嵌套通讯

前言 在前端开发项目中&#xff0c;不可避免的总会和 iframe 进行打交道&#xff0c;我们通常会使用 postMessage 实现消息通讯。 如果存在下面情况&#xff1a; iframe 父子通讯iframe 同层级通讯iframe 嵌套层级通讯 当面对这种复杂的情况的时候&#xff0c;通讯不可避免…

Uptime Kuma 使用指南:一款简单易用的站点监控工具

我平时的工作会涉及到监控&#xff0c;而站点是一个很重要的监控项。项目上线后&#xff0c;我们通常会将站点监控配置到云平台上&#xff0c;以检测各站点的连通性。但随着项目不断增多&#xff0c;云平台上的配额就有点捉急了。针对这个情况&#xff0c;我们可以试试这个开源…

李沐49_样式迁移——自学笔记

样式迁移 将样式图片中的样式迁移到内容图片上&#xff0c;合成图片&#xff0c;例如将照片转换成漫画形式或者是油画风。 基于CNN的样式迁移 读取图片和样式风格 %matplotlib inline import torch import torchvision from torch import nn from d2l import torch as d2ld…

Facebook的魅力魔法:探访数字社交的奇妙世界

1. 社交媒体的演变与Facebook的角色 在数字化时代&#xff0c;社交媒体已经成为我们日常生活中不可或缺的一部分。而在众多的社交媒体平台中&#xff0c;Facebook 以其深厚的历史和广泛的影响力&#xff0c;成为了全球数亿用户沟通、分享和互动的主要场所。从其初创之时起&…

【学习AI-相关路程-自我总结-相关入门-自我学习-NVIDIA-Jetson】

【学习AI-相关路程-自我总结-相关入门-自我学习】 1、前言2、思考前进方向3、学习路线1、基础知识阶段2、初级准备阶段3、中级学习阶段4、高级实战阶段 4、自我的努力5、学习平台6、自己总结 1、前言 最近AI相关比较火的&#xff0c;对于程序员&#xff0c;或者走这行的人来说…

Flutter开发好用插件url_launcher详解-启动 URL

文章目录 url_launcher介绍安装用法错误处理自定义行为其他功能 url_launcher介绍 url_launcher 是一个 Flutter 插件&#xff0c;用于启动 URL。它支持网络、电话、短信和电子邮件方案。您可以使用它从您的 Flutter 应用程序中打开网站、拨打号码、发送短信或撰写电子邮件。 …

群组分析方法

目录 1.什么是群组分析方法 2.基本原理 3.群组分析方法分类 3.1.层次方法 3.2.划分方法 3.3.密度基方法 ​​​​​​​3.4.模型基方法 4.群组评估 5.应用步骤 1.什么是群组分析方法 群组分析&#xff08;Cluster Analysis&#xff09;是数据分析中的一种重要方法&…

git lab 2.7版本修改密码命令

1.gitlab-rails console -e production Ruby: ruby 2.7.5p203 (2021-11-24 revision f69aeb8314) [x86_64-linux] GitLab: 14.9.0-jh (51fb4a823f6) EE GitLab Shell: 13.24.0 PostgreSQL: 12.7 2根据用户名修改密码 user User.find_by(username: ‘username’) # 替换’use…

ABAP 遗传算法求解

本文无文本解析&#xff0c;结尾处有简单装箱问题的示例&#xff0c;该算法收敛结果较慢&#xff0c;仅供ABAP爱好者参考&#xff0c;实践&#xff0c;实际应用建议使用线性规划。可直接复制后在系统中使用。 对象自定义逻辑版本-截图 对象自定义逻辑版本-对象描述 INIT I…

Navicat连接SQLSever报错:[08001] MicrosoftTCP Provider 远程主机强迫关闭了一个现有的连接

Navicat连接SQLSever报错&#xff1a;[08001] [Microsoft][SQL Server Native Client 10.0]TCP Provider: 远程主机强迫关闭了一个现有的连接 问题分析 旧版的MSSQL 如果不是最新版的&#xff0c;可以去这安装以下即可。 最新版的MSSQL 如果是安装最新版的MSSQL连接不上很正…

DFS和回溯专题:全排列 II

DFS和回溯专题&#xff1a;全排列 II 题目链接: 全排列 II 参考题解 代码随想录 题目描述 代码纯享版 class Solution {public List<List<Integer>> list_all new ArrayList();public List<Integer> list new ArrayList();public int[] res;public Lis…

探索 Python 的动态类型系统:变量引用、不可变性及高效内存管理与垃圾回收机制的深入分析

文章目录 1. 动态类型及其内存管理解析1.1 变量与对象的引用关系1.2 对象的不可变性和内存地址的变化 2. 垃圾回收与内存优化策略2.1 动态内存分配的基础2.2 Python 的垃圾回收 Python作为一种流行的高级编程语言&#xff0c;以其代码的易读性和简洁性著称。尤其是它的动态类型…

开源数据集分享———猫脸码客

猫脸码客作为一个专注于开源数据集分享的公众号&#xff0c;致力于为广大用户提供丰富、优质的数据资源。我们精心筛选和整理各类开源数据集&#xff0c;涵盖机器学习、深度学习、自然语言处理等多个领域&#xff0c;以满足不同用户的需求。 (https://img-blog.csdnimg.cn/d98…

找不到vcruntime140_1.dll,无法继续执行代码的多种解决方法

在启动电脑并着手进行日常工作的过程中&#xff0c;当我尝试运行一款至关重要的软件时&#xff0c;系统突然弹出一个令人困扰的错误提示&#xff1a;“由于找不到vcruntime140_1.dll&#xff0c;无法继续执行代码”&#xff0c;这个错误信息明确指出&#xff0c;由于缺失了vcru…

SpringCloud系列(16)--将服务提供者Provider注册进Zookeeper

前言&#xff1a;在上一章节中我们说明了一些关于Eureka自我保护模式&#xff0c;而且自上一章节起关于Eureka的知识已经讲的差不多了&#xff0c;不过因为Eureka已经停更了&#xff0c;为了安全考虑&#xff0c;我们要用还在更新维护的注册中心来取代Eureka&#xff0c;而本章…