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

 更新时间:2021年07月30日 10:47:39   作者:爱编辑  
这篇文章主要为大家详细介绍了C语言实现扫雷算法简易版,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

扫雷分析

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

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

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

为了解决数据越界的问题,我们最好创建一个比可视界面大一圈的数组,即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("\n");
 }
 
}

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

void DisplayBoard(char board[ROWS][COLS], int row, int col)
{
 int i = 0;
 int j = 0;
 for (i = 0; i <= col; i++)
 {
  printf("--");//将整个棋盘布局压在下面,可以当做分割线
 }
 printf("\n");
 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");
 }
 for (i = 0; i <= col; i++)
 {
  printf("--");//将整个棋盘布局顶在上面,可以当做分割线
 }
 printf("\n");
}

效果图:

放雷

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

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

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

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

效果图:

头文件:

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

游戏主干和菜单:

#include"game.h"
 
void menu()
{
 printf("***********************************************\n");
 printf("************           1.play     *************\n");
 printf("************           0.exit     *************\n");
 printf("***********************************************\n");
}
void game()
{
 printf(" >> >> >>>扫雷<<< << <<\n");
 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("您已退出游戏\n");
   break;
  default:
   printf("输入错误,请重新输入:>\n");
   break;
  }
 
 } while (input);
 return 0;
}

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

相关文章

  • C++编写DLL动态链接库的步骤与实现方法

    C++编写DLL动态链接库的步骤与实现方法

    这篇文章主要介绍了C++编写DLL动态链接库的步骤与实现方法,结合实例形式分析了C++导出类文件及生成与调用DLL动态连接库的相关操作技巧,需要的朋友可以参考下
    2016-08-08
  • C语言实现销售管理系统设计

    C语言实现销售管理系统设计

    这篇文章主要为大家详细介绍了C语言实现销售管理系统设计,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-03-03
  • C++ 先对数组排序,在进行折半查找

    C++ 先对数组排序,在进行折半查找

    以下小编就为大家介绍两种实现方法。第一种方法是,选择排序法+循环折半查找法。第二种方法是,冒泡排序法+递归折半查找法。需要的朋友可以过来参考下,希望对大家有所帮助
    2013-10-10
  • 节序问题:解析大小的端判定

    节序问题:解析大小的端判定

    本篇文章是对大小的端判定进行了详细的分析介绍,需要的朋友参考下
    2013-05-05
  • 减小VC6编译生成的exe文件的大小的方法

    减小VC6编译生成的exe文件的大小的方法

    这篇文章主要介绍了减小VC6编译生成的exe文件的大小的方法,需要的朋友可以参考下
    2015-01-01
  • C语言实现高精度减法

    C语言实现高精度减法

    高精度的本质是将数字以字符串的形式读入,然后将每一位分别存放入int数组中,通过模拟每一位的运算过程,来实现最终的运算效果,下面我们就来看看C语言如何实现高精度减法吧
    2023-11-11
  • 基于C语言实现的TCP服务器的流程分析

    基于C语言实现的TCP服务器的流程分析

    本文详细介绍了如何使用C语言编写一个简单的TCP服务器,包括创建套接字、绑定IP和端口、监听连接请求、接受客户端连接、数据接收与发送以及关闭套接字等步骤,最后通过一个简单的示例展示了TCP服务器的基本实现过程
    2024-10-10
  • C语言中动态内存管理初学者容易犯的6个错误分享

    C语言中动态内存管理初学者容易犯的6个错误分享

    本篇文章主要介绍了初学者使用C语言中动态内存管理的4个函数时最容易犯的6个错误,以及如何避免这些错误,文中的示例代码讲解详细,感兴趣的可以了解一下
    2023-04-04
  • C++栈(stack)的模板类实现代码

    C++栈(stack)的模板类实现代码

    这篇文章主要为大家详细介绍了C++栈(stack)的模板类实现代码,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-06-06
  • C++关于字符的接收与输出操作示例

    C++关于字符的接收与输出操作示例

    这篇文章主要介绍了C++关于字符的接收与输出操作,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习吧
    2023-01-01

最新评论