C语言游戏——扫雷

简介: C语言游戏——扫雷

一.菜单界面

这里的菜单与前文三字棋是一样的思路,通过(input)输入的值为条件,用switch或if来作分支判断,0为退出,1为开始游戏,其他不符则重新循环。

#include <stdio.h>
void menu()
{
  printf("********************\n");
  printf("*****  1.play  *****\n");
  printf("*****  0.exit  *****\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);
}

二.game函数

2.1 扫雷基本规则

我们可以假想所有的初始坐标都是0,布置雷的时候再把对应坐标变为1.

当我们要排雷的时候,假设先选上一个坐标,如果这个坐标不是雷的话看看周围8格是否有雷。如图:圈出来的坐标由0变1,因为周围8格里左边有一颗雷。

但是这样我们会发现这个查雷数字1与雷本身的数字1是一样的,会导致无法辨别。 这个时候我们可以再创造一个数组,这个数组专门用于给玩家看到的画面。

还有一个问题,当我们选中边界坐标时,数组以外的坐标又要怎么处理呢?

当然你可以选择在每一次的坐标中都遍历周围8个数组是否越界,不过这样太麻烦了。所以我们可以选择扩大数组范围,只要保证这个扩大的一圈中没有雷就可以解决问题了。

既然内部扩大数组了,那么玩家显示的数组也要跟着同步扩大,让坐标互相匹配。

为了数组类型的严格统一(方便用同一个调用函数),两个数组统一用字符,那么里面的数字也就变成了‘0’、‘1’。

下面开始定义:

2.2 初始化函数——InitBoard()

这里有一个小细节,为了达到两个数组都能初始化(因为0与*冲突),我们只需要再传一个对应初始化数值的参数就可以了。

2.3 显示棋盘函数——DisplayBoard()

因为这是来显示棋盘的,所以后面用到的范围都是row与col。其次,为了表示方便,添加了坐标轴来优化格式。

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

2.4 布置雷函数——SetMine()

因为坐标范围是1-9,所以只需要%row+1就可以定好范围了。关于随机函数rand()的用法,在猜数字一文中有具体介绍。唯一要注意的是要添加判断,因为如果随机的坐标重复的话,那么雷的数目就少了。

2.5 排查雷——FindMine()

排查雷我们需要传两个数组过去,因为它涉及到了2个数组。

在排雷函数中有几个需要注意的点:一是限制输入坐标,让坐标合法。二是当输入的坐标恰好是雷时,应该展现mine数组,让玩家知道情况。最后就是统计周围雷的数目,这个是最重要的。

void FindMine(char mine[ROWS][COLS],char show[ROWS][COLS], int row, int col)
{
  int x = 0;
  int y = 0;
  while (1)
  {
    printf("请输入排雷坐标:>");
    scanf("%d %d", &x, &y);
    if (1 <= x && x <= 9 && 1 <= y && y <= 9)
    {
      if (mine[x][y] == '1')
      {
        printf("很遗憾,你被炸死了\n");
        DisplayBoard(mine, ROW, COL);
        break;
      }
      else
      {
        //排雷,统计坐标周围有几个雷
      }
    }
    else
    {
      printf("输入错误,请重新输入:\n");
    }
  }
}

这是假想坐标中周围坐标的规律:

我们可以想到可以通过让周围8个坐标全部相加来得到雷数,但需要注意的是,这些都是字符而非数字。所以让每一个字符都减去‘0’,再相加就可以得到数字了。

2.6 统计雷——GetMineCount()

剩下的统计就很简单了,依次相加即可,然后每一次统计完成后再显示一次玩家数组就完成了。

void FindMine(char mine[ROWS][COLS],char show[ROWS][COLS], int row, int col)
{
  int x = 0;
  int y = 0;
  while (1)
  {
    printf("请输入排雷坐标:>");
    scanf("%d %d", &x, &y);
    if (1 <= x && x <= 9 && 1 <= y && y <= 9)
    {
      if (mine[x][y] == '1')
      {
        printf("很遗憾,你被炸死了\n");
        DisplayBoard(mine, ROW, COL);
        break;
      }
      else
      {
        //排雷,统计坐标周围有几个雷
        int c = GetMineCount(mine, x, y);
        show[x][y] = c + '0';
        DisplayBoard(show, ROW, COL);
      }
    }
    else
    {
      printf("输入错误,请重新输入:\n");
    }
  }
}
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';
}

接下来是测试阶段:

这里还有一点,就是还没有设立雷全排出来的胜利阶段;在9*9中有10个雷,意味着需要排71次才可以排完,所以可以设立一个条件win,没排满71次就继续循环,每次查完自增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 (1 <= x && x <= 9 && 1 <= y && y <= 9)
    {
      if (mine[x][y] == '1')
      {
        printf("很遗憾,你被炸死了\n");
        DisplayBoard(mine, ROW, COL);
        break;
      }
      else
      {
        //排雷,统计坐标周围有几个雷
        int c = GetMineCount(mine, x, y);
        show[x][y] = c + '0';
        DisplayBoard(show, ROW, COL);
                win++;
      }
    }
    else
    {
      printf("输入错误,请重新输入:\n");
    }
  }
  if (win = row * col - EASY_COUNT)
  {
    printf("恭喜你,排雷成功\n");
    DisplayBoard(mine, ROW, COL);
  }
}

在快速排查阶段,我们把雷改成80个试一下。

三.扫雷代码优化

我们的扫雷都是一步一步扫的,过于低效,那么可以做到想网上游戏一样扫出一片吗?

 

首先看坐标是不是雷,如果坐标不是雷再看坐标周围8个格是不是雷,以此作为循环。——递归

但是递归会带有太多的重复计算,所以当确定不是雷后可以用空格来标记此处。

关于这个优化,我后续会给出我的解答,在这里也期待大家对这个问题的观点与看法,谢谢。

相关文章
|
11天前
|
存储 C语言 开发者
C语言实战 | Flappy Bird游戏
【7月更文挑战第4天】Flappy Bird是由越南开发者制作的简单却极具挑战性的游戏,玩家需控制小鸟穿越水管障碍。游戏涉及角色初始化、显示和更新。小鸟和水管结构体存储数据,使用变量和数组。初始化小鸟和水管,显示背景、小鸟和水管,更新小鸟位置及碰撞检测。代码示例展示了小鸟和水管的状态管理,当小鸟与管道碰撞或触地时,游戏结束。游戏的成功在于其独特的虐心体验。
21 0
C语言实战 | Flappy Bird游戏
|
9天前
|
存储 编译器 C语言
|
5天前
|
存储 编译器 C语言
C语言实战 | “贪吃蛇”游戏
【7月更文挑战第5天】在C语言实战中,本文档介绍了如何构建一个简单的“贪吃蛇”游戏。游戏的核心是控制蛇移动并增长,当吃掉食物时,蛇的身体变长。数据结构使用固定大小的数组表示蛇的位置,变量存储食物位置和蛇的长度。初始化后,利用非阻塞式`getKey()`函数实现WASD键盘控制蛇的运动方向。虽然蛇的边界检测和吃食物后的增长尚未详细说明,但提到了这些问题作为练习留给读者解决,并预告将在后续章节讨论模块化编程以简化复杂代码。
16 0
C语言实战 | “贪吃蛇”游戏
|
13天前
|
存储 数据管理 C语言
C语言实战 | 使用链表完成“贪吃蛇”游戏
【7月更文挑战第1天】整体思维,即系统思维,强调以整体视角理解事物。在编程中,结构体体现这种思想,将相关变量打包处理。示例展示了如何用链表而非数组实现“贪吃蛇”游戏,链表提供了更灵活的动态数据管理。一系列代码图片详细描绘了链表结构体在游戏中的应用,包括节点定义、移动、碰撞检测等,凸显了使用链表的优势和代码的清晰组织。
17 0
C语言实战 | 使用链表完成“贪吃蛇”游戏
|
20天前
|
算法 编译器 C语言
猜数字游戏C语言代码实现
猜数字游戏C语言代码实现
|
18天前
|
存储 C语言
C语言实战 | “贪吃蛇”游戏重构
在程序设计中,模块化思维至关重要,尤其对于复杂项目,它帮助分解任务,便于团队协作。以“贪吃蛇”游戏为例,游戏涉及两个角色:蛇和食物。使用数组存储蛇的位置,变量存储食物位置。游戏流程分为初始化、显示和更新数据。初始化时,食物位置随机,蛇的位置根据数组设定。显示数据则根据这些信息在屏幕上呈现角色。更新数据时,处理蛇的移动和增长以及食物的生成和消失。类似地,通过模块化方法可开发“打砖块”游戏,涉及球、球拍和砖墙,每个角色都有相应数据结构和更新逻辑。通过这种方式,游戏开发就像搭建积木,遵循框架逐步实现。
23 0
C语言实战 | “贪吃蛇”游戏重构
|
20天前
|
C语言
【海贼王编程冒险 - C语言海上篇】怎样用C语言实现简单的扫雷游戏?
【海贼王编程冒险 - C语言海上篇】怎样用C语言实现简单的扫雷游戏?
10 1
|
20天前
|
C语言
【海贼王编程冒险 - C语言海上篇】C语言如何实现简单的三子棋游戏?
【海贼王编程冒险 - C语言海上篇】C语言如何实现简单的三子棋游戏?
11 1
|
20天前
|
存储 安全 Serverless
扫雷游戏C语言代码实现——万字长文超详细,手把手教你实现,新手也能学会
扫雷游戏C语言代码实现——万字长文超详细,手把手教你实现,新手也能学会
|
21天前
|
存储 定位技术 API
C语言实战 -- 经典贪吃蛇游戏(含完整源码)
C语言实战 -- 经典贪吃蛇游戏(含完整源码)
20 1