【熟视C语言】扫雷——C语言练习项目,一起锻炼代码能力

2024-03-26 13:40

本文主要是介绍【熟视C语言】扫雷——C语言练习项目,一起锻炼代码能力,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

前言

设计思路

游戏菜单

游戏模块

Initi_board初始化方阵内元素

display_board展示二维数组的内容

set_mine传入mine数组设置地雷

check_mine排查地雷

Total返回坐标四周8个坐标中地雷的个数

Expand从传入坐标出发排查方阵中周围无地雷的位置直至有雷的位置

小结

彩蛋


前言

  本篇文章将带你使用C语言编写小游戏扫雷,来跟着试着一起思考和编写,锻炼你的代码能力吧。

设计思路

  我们设想需要设计的功能有菜单界面,进入游戏生成9×9的扫雷游戏区域,使用坐标确定排查位置,以及当排查到四周无雷的时候自动展开的功能,当玩家输入到藏雷的坐标时,则游戏结束玩家失败,当玩家排查出所有非雷坐标时,游戏结束,玩家胜利。

游戏菜单

  这部分内容比较简单,也好实现,循环体如果愿意也可不使用我的dowhile语句和switch语句,可自行设计,建议设计思路以简单为主,方便维护或迭代。

void menu()
{printf("*********************************\n");printf("*******      1.play        ******\n");printf("*******      0.exit        ******\n");printf("*********************************\n");
}int main()//扫雷
{int input = 0;do{menu();scanf_s("%d", &input);switch (input){case 1:printf("        扫雷\n");game();break;case 0:printf("退出游戏\n");break;default:printf("选择错误,请重新选择!\n");break;}} while (input);return 0;
}

游戏模块

  这个函数用于确定游戏主体框架,运行思路,将各个功能模块以及数据整和在一起,便于阅读,维护以及更新迭代。(此处二维数组的创建使用两个宏,ROWS和COLS,值都是11,至于为什么创建11×11的方阵后面会讲解)除此之外,这里有一点比较重要的是使用srand函数设置rand函数的起点(用于随机生成地雷的坐标)。接下来我会细致的讲讲这几个函数或者说模块是如何实现的。

void game()//游戏模块实现
{char mine[ROWS][COLS];char show[ROWS][COLS];srand((unsigned int)time(NULL));//初始化排雷方阵Initi_board(mine, ROWS, COLS, '0');Initi_board(show, ROWS, COLS, '*');//显示方阵//display_board(mine, ROWS, COLS);display_board(show, ROWS, COLS);//布置雷set_mine(mine, ROWS, COLS, MINE);//display_board(mine, ROWS, COLS);//排查雷check_mine(mine, show, ROWS, COLS);}

Initi_board初始化方阵内元素

  由于我们两个方阵需要初始化的字符不一样,所以传参时需要的参数除了,二维数组,方阵的行和列,还需要我们需要初始化的字符。(此部分比较简单,就不放入思维导图了)

void Initi_board(char board[ROWS][COLS], int row, int col, char ch)
{int i = 0;int j = 0;for (i = 0; i < row; i++){for (j = 0; j < col; j++){board[i][j] = ch;}}}

display_board展示二维数组的内容

  这部分需要注意的是,我们需要打印的是9×9方阵,而我们创建的数组是11×11的,所以打印时时使用的下标是1至9。(至于为什么创建11×11的二维数组我会在下面解释,同样,此部分比较简单,就放入思维导图了)

void display_board(char board[ROWS][COLS], int row, int col)
{int i = 0;int j = 0;for (i = 0; i < col - 1; i++){printf("%d ", i);}printf("\n");for (i = 1; i < row - 1; i++){printf("%d ", i);for (j = 1; j < col - 1; j++){printf("%c ", board[i][j]);}printf("\n");}
}

set_mine传入mine数组设置地雷

  这部分主要的问题是如何得到两个在1到9之间的随机数来当作地雷的坐标,而我们只需要使用rand函数就可以解决这个问题。rand函数会相对于一个起点随机生成一个数,而为了这个随机数在1到9之间,我们只需要对这个数摸9加1就可以将此数变成1到9之间的数字了。除此之外,此函数的传参还额外需要一个整型数据,该数据是地雷的数量,在game函数中使用了自定义的宏MINE,该宏的值是10,这是用于便于将来更新此游戏的地雷数量的。

void set_mine(char board[ROWS][COLS], int row, int col, int m)
{while (m){int x = rand() % 9 + 1;int y = rand() % 9 + 1;if (board[x][y] == '0'){board[x][y] = '1';m--;}}
}

check_mine排查地雷

  这部分有几点需要注意,第一是count的赋值使用宏ROW×COL-MINE,其中ROW与COL的值是9,MINE的值是10,便于以后更新迭代,第二点是需要判断输入坐标是否合法,避免出现bug,第三点是Total函数对应功能的实现以及坐标对应元素数据的修改方式,这点我会在接下来Tatol函数讲解中说明,第四点是设计或编写代码时注意count是否能控制循环或者控制游戏是否结束。(当你试着对这个游戏更新迭代添加新功能时需要格外注意,例如增加插旗功能时)

void check_mine(char board1[ROWS][COLS], char board2[ROWS][COLS], int row, int col)
{int x = 0;int y = 0;int count = ROW * COL - MINE;while (count){printf("输入您想检查的坐标:\n");scanf("%d %d", &x, &y);if (x >= 1 && x < row && y >= 1 && y < col){if (board1[x][y] == '1'){printf("很遗憾,您已死亡,游戏结束\n");display_board(board1, ROWS, COLS);break;}else{board2[x][y] = Total(board1, x, y) + '0';if (board2[x][y] == '0')Expand(board1, board2, x, y, &count);display_board(board2, ROWS, COLS);//display_board(board1, ROWS, COLS);count--;}}else{printf("抱歉,您输入的坐标不合法,请重新输入\n");}}if (count == 0){printf("恭喜你排查出所有的雷,游戏胜利\n");display_board(board1, ROWS, COLS);printf("是否进行下一局:\n");}}

Total返回坐标四周8个坐标中地雷的个数

  在解释将宏ROWS和COLS的值设为11或者说创建11×11方阵的原因之前,我先讲一下Total函数的设计思路,由于我们只需要计算传入坐标的四周的雷也就是字符‘1‘的数量,所以我们只需要将这周围八个坐标的元素的值加起来,然后需要注意的一点是元素的类型是char类型,直接相加并不能得到我们想要的数据,因此需要减上8×’0‘,也就是直接返回传入坐标周围八个元素相加的值减去8×’0‘,除此之外,我们还需要在使用返回值时对返回值加上一个’0‘以确保此时的数据是我们想要的那个数值。(此处如不能理解请参考ASCII表)

  现在,为什么mine要创建成11×11的方阵的原因已经显而易见了,因为我的函数设计是直接返回周围8个数据的运算,但是如果是在跟游戏需要方阵一样大的9×9方阵中,在某些位置周围是没有八个数据,这时还需要我们在该函数判断是否处于特殊位置,那么2此时代码设计则过于繁琐低效。(至于为什么是char类型,其实int类型也行,但此时返回值也是int类型,对返回值需要进行强制类型转换,读者可自行编写实现出自己的想法)

static char Total(char board[ROWS][COLS], int row, int col)
{return board[row - 1][col - 1] +board[row][col - 1] +board[row + 1][col - 1] +board[row + 1][col] +board[row + 1][col + 1] +board[row][col + 1] +board[row - 1][col + 1] +board[row - 1][col] - 8 * '0';
}

Expand从传入坐标出发排查方阵中周围无地雷的位置直至有雷的位置

  为保证count可用于判断玩家胜利(控制循环),这里的传参需要将count的地址传过来以便使用和修改,在后续使用中,在修改show数组中内容时需将count内容一起修改。初次之外,需要对传入坐标对于show数组的值进行判断是否为‘*’以防重复调用Expand而出现死递归。这两点是设计这个函数是需要注意的地方。

void Expand(char board_a[ROWS][COLS], char board_b[ROWS][COLS], int rows, int cols, int * p)//从传入坐标出发排查方阵中周围无地雷的位置直至有雷的位置
{if (rows < ROW && (Total(board_a, rows + 1, cols) + '0') == '0' && (board_b[rows + 1][cols] != '0')){board_b[rows + 1][cols] = '0';Expand(board_a, board_b, rows + 1, cols, p);(*p)--;}else if (rows < ROW && (Total(board_a, rows + 1, cols) + '0') > '0' && (board_b[rows + 1][cols] != '0')){board_b[rows + 1][cols] = Total(board_a, rows + 1, cols) + '0';(*p)--;}if (rows > 1 && (Total(board_a, rows - 1, cols) + '0') == '0' && (board_b[rows - 1][cols] != '0')){board_b[rows - 1][cols] = '0';Expand(board_a, board_b, rows - 1, cols, p);(*p)--;}else if (rows > 1 && (Total(board_a, rows - 1, cols) + '0') > '0' && (board_b[rows - 1][cols] != '0')){board_b[rows - 1][cols] = Total(board_a, rows - 1, cols) + '0';(*p)--;}if (cols < COL && (Total(board_a, rows, cols + 1) + '0') == '0' && (board_b[rows][cols + 1] != '0')){board_b[rows][cols + 1] = '0';Expand(board_a, board_b, rows, cols + 1, p);(*p)--;}else if (cols < COL && (Total(board_a, rows, cols + 1) + '0') > '0' && (board_b[rows][cols + 1] != '0')){board_b[rows][cols + 1] = Total(board_a, rows, cols + 1) + '0';(*p)--;}if (cols > 1 && (Total(board_a, rows, cols - 1) + '0') == '0' && (board_b[rows][cols - 1] != '0')){board_b[rows][cols - 1] = '0';Expand(board_a, board_b, rows, cols - 1, p);(*p)--;}else if (cols > 1 && (Total(board_a, rows, cols - 1) + '0') > '0' && (board_b[rows][cols - 1] != '0')){board_b[rows][cols - 1] = Total(board_a, rows, cols - 1) + '0';(*p)--;}
}

小结

  本篇博客讲解的扫雷游戏主要使用C语言实现,所以操作界面比较简陋,所用到的知识也比较简单,有二维数组,函数调用,递归等,适合C语言初学者锻炼基础的项目,如果想开发一个完备的扫雷游戏可在游戏引擎上设计此游戏,设计思路可参照本代码。

彩蛋

  以下展示我的代码文件配置,我也会将代码文件上传至我的线上代码库,有需要自取。

gitee:https://gitee.com/crazy-little-confucian/c-language-learning.git  test分支文件名为扫雷

github:https://github.com/crrrush/c-language.git  test分支文件名为扫雷

 

 

这篇关于【熟视C语言】扫雷——C语言练习项目,一起锻炼代码能力的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

C++使用栈实现括号匹配的代码详解

《C++使用栈实现括号匹配的代码详解》在编程中,括号匹配是一个常见问题,尤其是在处理数学表达式、编译器解析等任务时,栈是一种非常适合处理此类问题的数据结构,能够精确地管理括号的匹配问题,本文将通过C+... 目录引言问题描述代码讲解代码解析栈的状态表示测试总结引言在编程中,括号匹配是一个常见问题,尤其是在

部署Vue项目到服务器后404错误的原因及解决方案

《部署Vue项目到服务器后404错误的原因及解决方案》文章介绍了Vue项目部署步骤以及404错误的解决方案,部署步骤包括构建项目、上传文件、配置Web服务器、重启Nginx和访问域名,404错误通常是... 目录一、vue项目部署步骤二、404错误原因及解决方案错误场景原因分析解决方案一、Vue项目部署步骤

Java调用DeepSeek API的最佳实践及详细代码示例

《Java调用DeepSeekAPI的最佳实践及详细代码示例》:本文主要介绍如何使用Java调用DeepSeekAPI,包括获取API密钥、添加HTTP客户端依赖、创建HTTP请求、处理响应、... 目录1. 获取API密钥2. 添加HTTP客户端依赖3. 创建HTTP请求4. 处理响应5. 错误处理6.

python使用fastapi实现多语言国际化的操作指南

《python使用fastapi实现多语言国际化的操作指南》本文介绍了使用Python和FastAPI实现多语言国际化的操作指南,包括多语言架构技术栈、翻译管理、前端本地化、语言切换机制以及常见陷阱和... 目录多语言国际化实现指南项目多语言架构技术栈目录结构翻译工作流1. 翻译数据存储2. 翻译生成脚本

使用 sql-research-assistant进行 SQL 数据库研究的实战指南(代码实现演示)

《使用sql-research-assistant进行SQL数据库研究的实战指南(代码实现演示)》本文介绍了sql-research-assistant工具,该工具基于LangChain框架,集... 目录技术背景介绍核心原理解析代码实现演示安装和配置项目集成LangSmith 配置(可选)启动服务应用场景

golang内存对齐的项目实践

《golang内存对齐的项目实践》本文主要介绍了golang内存对齐的项目实践,内存对齐不仅有助于提高内存访问效率,还确保了与硬件接口的兼容性,是Go语言编程中不可忽视的重要优化手段,下面就来介绍一下... 目录一、结构体中的字段顺序与内存对齐二、内存对齐的原理与规则三、调整结构体字段顺序优化内存对齐四、内

Python中顺序结构和循环结构示例代码

《Python中顺序结构和循环结构示例代码》:本文主要介绍Python中的条件语句和循环语句,条件语句用于根据条件执行不同的代码块,循环语句用于重复执行一段代码,文章还详细说明了range函数的使... 目录一、条件语句(1)条件语句的定义(2)条件语句的语法(a)单分支 if(b)双分支 if-else(

Go语言中三种容器类型的数据结构详解

《Go语言中三种容器类型的数据结构详解》在Go语言中,有三种主要的容器类型用于存储和操作集合数据:本文主要介绍三者的使用与区别,感兴趣的小伙伴可以跟随小编一起学习一下... 目录基本概念1. 数组(Array)2. 切片(Slice)3. 映射(Map)对比总结注意事项基本概念在 Go 语言中,有三种主要

MySQL数据库函数之JSON_EXTRACT示例代码

《MySQL数据库函数之JSON_EXTRACT示例代码》:本文主要介绍MySQL数据库函数之JSON_EXTRACT的相关资料,JSON_EXTRACT()函数用于从JSON文档中提取值,支持对... 目录前言基本语法路径表达式示例示例 1: 提取简单值示例 2: 提取嵌套值示例 3: 提取数组中的值注意

配置springboot项目动静分离打包分离lib方式

《配置springboot项目动静分离打包分离lib方式》本文介绍了如何将SpringBoot工程中的静态资源和配置文件分离出来,以减少jar包大小,方便修改配置文件,通过在jar包同级目录创建co... 目录前言1、分离配置文件原理2、pom文件配置3、使用package命令打包4、总结前言默认情况下,