服务器之家:专注于VPS、云服务器配置技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - C/C++ - C语言实现扫雷算法简易版

C语言实现扫雷算法简易版

2021-12-08 13:25爱编辑 C/C++

这篇文章主要为大家详细介绍了C语言实现扫雷算法简易版,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

扫雷分析

从小到大你或许没玩过但一定听过的游戏――扫雷

首先我们来分一下“扫雷”的功能

C语言实现扫雷算法简易版

这是一个简单难度的扫雷,从外观上,我们可以发现可供用户操作的棋盘范围是9×9的范围,也就是我们建立的棋盘大小至少要为9,但是问题也就来了,我们如果只建立9×9的棋盘,那么在边缘的格子要进行提示操作的时候就会出现数据越界问题。

C语言实现扫雷算法简易版

为了解决数据越界的问题,我们最好创建一个比可视界面大一圈的数组,即11×11的数组

但是在我们开始做这个小游戏的时候发现了一个问题――一个数组里面包含的数据太多了,可能会发生数据重叠的现象,而且一个数组里面又要放雷,又要存放排雷之后的信息,还要遮掩雷和其他位置,太过麻烦,因此我们可以创建两个数组,一个数组专门用来放雷,一个数组用来存放排雷的信息,注意,这两个数组一定要保证大小一样。

所以我们不妨设置这样两个数组

mine[11][11]  //存放雷的信息
show[11][11]  //存放排除的雷的信息
/*如果我们直接用数字11来初始化数组,局限性太大了,我们后面如果想要
**进行修改也不好改,所以我们可以引用一个全局变量来初始化数组,这样
**即使我们以后想要玩更大的棋盘扫雷,就能够做到一步更改。考虑到棋盘
**数组的初始化要比棋盘大一圈,所以可以设置为下面这样。
*/
 
#define ROW 9  //可视化界面是9×9
#define COL 9
#define ROWS ROW+2
#define COLS COL+2
mine[ROWS][COLS]
show[ROWS][COLS]
 
//为了和扫雷游戏保持一样我们在显示扫雷棋盘的时候用符号将它遮住,这里用的是“*”,所以数组类型就确定了
char mine[ROWS][COLS]
char show[ROWS][COLS]

棋盘初始化

建立好数组之后,我们就开始思考可视化界面的建立,根据前面的信息,我们要创建一个9×9的游戏空间时得扩大一圈,因此创建的函数传参用ROWS和COLS,并且我们对这两个的数组初始化也要做出差异,那么要如何来保证一个函数能同时初始化2个数组呢?实现如下:

//将数组初始化的信息作为区分参数,传入函数即可
 
InitBoard(mine, ROWS, COLS, "0");//没有雷的地方存放字符‘0"
InitBoard(show, ROWS, COLS, "*");//‘*"遮掩扫雷棋盘
 
//接收传参的函数
void InitBoard(char board[ROWS][COLS], int rows, int cols,char set)
{
 int i = 0;
 int j = 0;
 for (i = 0; i < rows; i++)
 {
  for (j = 0; j < cols; j++)
  {
   board[i][j] = set ;
  }
 }
 
}

棋盘显示

初始化之后,是骡子是马,拉出来溜溜,因为我们要保证数据不能越界所以可视化界面比实际数组小一圈

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

这种显示乍一看没什么问题,但是当你要进行输入的时候,你就难受了,因为你不知道它的行和列,每次输入都要去数一次,所以,我们可以进行优化

void DisplayBoard(char board[ROWS][COLS], int row, int col)
{
 int i = 0;
 int j = 0;
 for (i = 0; i <= col; i++)
 {
  printf("--");//将整个棋盘布局压在下面,可以当做分割线
 }
 printf("
");
 for (i = 0; i <= col; i++)
 {
  printf("%d ", i);//显示列的号数
 }
 printf("
");
 for (i = 1; i <= row; i++)
 {
  printf("%d ", i);//显示行的号数
  for (j = 1; j <= col; j++)
  {
   printf("%c ",board[i][j]);
  }
  printf("
");
 }
 for (i = 0; i <= col; i++)
 {
  printf("--");//将整个棋盘布局顶在上面,可以当做分割线
 }
 printf("
");
}

效果图:

C语言实现扫雷算法简易版

放雷

当扫雷棋盘可以正常显示出来之后,我们就可以开始做设置雷了

在设置雷的时候我们需要考虑,设置多少个雷,来形成简单,普通,困难等难度

因此,雷的数量在以后可能会发生变化,我们也可以将它设置为全局变量

#define easy_count 10

后面我们在放置雷的时候要确保它的随机性,因此需要用到rand()函数和time()函数

void SetMine(char board[ROWS][COLS], int row, int col)
{
 int count = EASY_COUNT;
 while (count)
 {
  int x = rand() % row + 1;
  int y = rand() % col + 1;
  if (board[x][y] != "1")/*确保了它只有在不是‘1"的空位上去放置雷,这样一来就可以
                                *避免数据覆盖而导致雷的数量不够*/
  {
   board[x][y] = "1";
   count--;
  }
 }
}

排雷和判定胜负

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 (x >= 1 && x <= row && y >= 1 && y <= col)
  {
   if (mine[x][y] == "1")
   {
    printf("很遗憾,你被炸死了!
");
    DisplayBoard(mine, ROW, COL);
    break;
   }
   else
   {
    
     int count = GetMineCount(mine, x, y);
     show[x][y] = count + "0";
     DisplayBoard(show, ROW, COL);
     win++;
   }
  }
  else
  {
   printf("非法坐标,请重新输入:>
");
  }
  
 }
 if (win == row * col - EASY_COUNT)
 {
  printf("恭喜你,排雷成功!
");
 }
}

效果图:

C语言实现扫雷算法简易版

头文件:

#pragma once
#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 79
//初始化棋盘
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 board[ROWS][COLS], int row, int col);
//排雷
void FindMine(char mine[ROWS][COLS], char show[ROWS][COLS], int row, int col);

游戏功能实现: 

#include"game.h"
 
void InitBoard(char board[ROWS][COLS], int rows, int cols,char set)
{
 int i = 0;
 int j = 0;
 for (i = 0; i < rows; i++)
 {
  for (j = 0; j < cols; j++)
  {
   board[i][j] = set ;
  }
 }
 
}
 
void DisplayBoard(char board[ROWS][COLS], int row, int col)
{
 int i = 0;
 int j = 0;
 for (i = 0; i <= col; i++)
 {
  printf("--");
 }
 printf("
");
 for (i = 0; i <= col; i++)
 {
  printf("%d ", i);
 }
 printf("
");
 for (i = 1; i <= row; i++)
 {
  printf("%d ", i);
  for (j = 1; j <= col; j++)
  {
   printf("%c ",board[i][j]);
  }
  printf("
");
 }
 for (i = 0; i <= col; i++)
 {
  printf("--");
 }
 printf("
");
}
 
void SetMine(char board[ROWS][COLS], int row, int col)
{
 int count = EASY_COUNT;
 while (count)
 {
  int x = rand() % row + 1;
  int y = rand() % col + 1;
  if (board[x][y] != "1")
  {
   board[x][y] = "1";
   count--;
  }
 }
}
 
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";
}
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 (x >= 1 && x <= row && y >= 1 && y <= col)
  {
   if (mine[x][y] == "1")
   {
    printf("很遗憾,你被炸死了!
");
    DisplayBoard(mine, ROW, COL);
    break;
   }
   else
   {
     int count = GetMineCount(mine, x, y);
      show[x][y] = count + "0";
     DisplayBoard(show, ROW, COL);
     win++;
    
   }
  }
  else
  {
   printf("非法坐标,请重新输入:>
");
  }
  
 }
 if (win == row * col - EASY_COUNT)
 {
  printf("恭喜你,排雷成功!
");
 }
}

游戏主干和菜单:

#include"game.h"
 
void menu()
{
 printf("***********************************************
");
 printf("************           1.play     *************
");
 printf("************           0.exit     *************
");
 printf("***********************************************
");
}
void game()
{
 printf(" >> >> >>>扫雷<<< << <<
");
 char mine[ROWS][COLS] = {0};
 char show[ROWS][COLS] = {0};
 InitBoard(mine, ROWS, COLS, "0");
 InitBoard(show, ROWS, COLS, "*");
 
 SetMine(mine,ROW,COL);
 DisplayBoard(mine, ROW, COL);
 DisplayBoard(show, ROW, COL);
 FindMine(mine,show,ROW,COL);
}
int main()
{
 int input;
 srand((unsigned int)time(NULL));
 do
 {
  menu();
  printf("请做出选择:>");
  scanf("%d", &input);
  switch (input)
  {
  case 1:
   game();
   break;
  case 0:
   printf("您已退出游戏
");
   break;
  default:
   printf("输入错误,请重新输入:>
");
   break;
  }
 
 } while (input);
 return 0;
}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持服务器之家。

原文链接:https://blog.csdn.net/m0_60117108/article/details/119189464

延伸 · 阅读

精彩推荐
  • C/C++深入C++拷贝构造函数的总结详解

    深入C++拷贝构造函数的总结详解

    本篇文章是对C++中拷贝构造函数进行了总结与介绍。需要的朋友参考下...

    C++教程网5182020-11-30
  • C/C++使用C++制作简单的web服务器(续)

    使用C++制作简单的web服务器(续)

    本文承接上文《使用C++制作简单的web服务器》,把web服务器做的功能稍微强大些,主要增加的功能是从文件中读取网页并返回给客户端,而不是把网页代码...

    C++教程网5492021-02-22
  • C/C++OpenCV实现拼接图像的简单方法

    OpenCV实现拼接图像的简单方法

    这篇文章主要为大家详细介绍了OpenCV实现拼接图像的简单方法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...

    iteye_183805102021-07-29
  • C/C++C语言main函数的三种形式实例详解

    C语言main函数的三种形式实例详解

    这篇文章主要介绍了 C语言main函数的三种形式实例详解的相关资料,需要的朋友可以参考下...

    ieearth6912021-05-16
  • C/C++关于C语言中E-R图的详解

    关于C语言中E-R图的详解

    今天小编就为大家分享一篇关于关于C语言中E-R图的详解,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看...

    Struggler095962021-07-12
  • C/C++C语言实现双人五子棋游戏

    C语言实现双人五子棋游戏

    这篇文章主要为大家详细介绍了C语言实现双人五子棋游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...

    两片空白7312021-11-12
  • C/C++c/c++内存分配大小实例讲解

    c/c++内存分配大小实例讲解

    在本篇文章里小编给大家整理了一篇关于c/c++内存分配大小实例讲解内容,有需要的朋友们可以跟着学习参考下。...

    jihite5172022-02-22
  • C/C++c/c++实现获取域名的IP地址

    c/c++实现获取域名的IP地址

    本文给大家汇总介绍了使用c/c++实现获取域名的IP地址的几种方法以及这些方法的核心函数gethostbyname的详细用法,非常的实用,有需要的小伙伴可以参考下...

    C++教程网10262021-03-16