C语言实现排雷游戏(多文件)

 更新时间:2020年07月17日 12:18:40   作者:Tianzez  
这篇文章主要为大家详细介绍了C语言实现排雷游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

本文实例为大家分享了C语言实现排雷游戏的具体代码,供大家参考,具体内容如下

游戏功能:

①打印雷盘
②随机布雷
③第一踩雷不死(重新布雷)
④扩展式扫雷
⑤计算周围雷的个数

代码关键点:

①玩游戏的雷盘比实际定义的数组小。
②memset初始化数组(以字节为单位初始化)。
③rand函数给雷盘随机位置布雷(用sand设置随机数种子)。
④第一步就踩到雷要这个雷移开,给玩家一次机会。
⑤踩到雷后根据情况决定是否进行扩展式排雷。
⑥返回输入排雷位置周围雷的个数,根据雷的数量决定是否进行扩展式排雷。
⑦如果输入的坐标周围有雷,那就不进行扩展式排雷,直接在该位置输出它周围雷的个数。
⑧输入的坐标周围无雷,进行扩展式排雷。
⑨注意扩展式排雷函数的形参,以及函数递归对形参的影响。
⑩注意进行函数递归的判断条件。

1.game.h

#ifndef __GAME_H__
#define __GAME_H__

#define COLS 11
#define ROWS 11
#define MAX 10

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <time.h>
#include <windows.h>

void Display(char arr[COLS][ROWS], int col, int row);
void set_mine(char mine[COLS][ROWS], int col, int row);
int get_mine_count(char mine[COLS][ROWS],int x,int y);
void reset_mine(char mine[COLS][ROWS], int col, int row, int x, int y);
void extend(char mine[COLS][ROWS], int x, int y, int* win, char show[COLS][ROWS]);

#endif //__GAME_H__

2.game.c

#include "game.h"

void Display(char arr[COLS][ROWS], int col, int row) //打印雷盘
{
  int i = 0;
  int j = 0;
  printf(" ");
  for (i = 1; i < col-1; ++i) //打印显示行坐标提示
  {
    printf("%2d ", i);
  }

  printf("\n");

  for (i = 1; i < col - 1; ++i)
  {
    printf("%d", i); //打印显示列坐标提示
    for (j = 1; j < row - 1; ++j)
    {
      printf("%2c ", arr[i][j]);
    }
    printf("\n");
  }
}

void set_mine(char mine[COLS][ROWS], int col, int row) //设置雷
{

  int count = MAX;
  while (count)
  {
    int i = rand() % (col - 2) + 1;
    int j = rand() % (row - 2) + 1;
    if (mine[i][j] != '1')
    {
      mine[i][j] = '1';
      count--;
    }
  }
}

void reset_mine(char mine[COLS][ROWS], int col, int row, int x, int y) //重新设置雷
{
  mine[x][y] = '0';
  int count = 1;
  while (count)
  {
    int i = rand() % (col - 2) + 1;
    int j = rand() % (row - 2) + 1;
    if ((mine[i][j] != '1') && (i != x) && (j != y))
    {
      mine[i][j] = '1';
      count--;
    }
  }
}

int get_mine_count(char mine[COLS][ROWS], int x, int y) //计算坐标(x,y)周围八个位置雷的个数并以int型返回
{
  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 extend(char mine[COLS][ROWS], int x, int y, int* win, char show[COLS][ROWS]) //排雷扩展
{
  int i = -1;
  int j = -1;
  for (i = -1; i < 2; ++i)
  {
    for (j = -1; j < 2; ++j)
    {
      if ((i != 0) || (j != 0))  
      {
        if (x + i >= 1 && x + i <= COLS-2 && y + j >= 1 && y + j <= ROWS-2) 
        {
          if (show[x + i][y + j] == '*')
          {
            int count = get_mine_count(mine, x + i, y + j);
            if (count != 0)
            {
              show[x + i][y + j] = count + '0';
              (*win)++;
            }
            else if (0 == count)
            {
              show[x + i][y + j] = '0';
              (*win)++;
              extend(mine, x + i, y + j, win, show);
            }
          }
        }
      }
    }
  }
}

3.test.c

#include "game.h"


void game()
{
  int x = 0;
  int y = 0;
  int win = 0;  //记录找到非雷位置的次数
  srand((unsigned int)time(NULL));       //随机数种子
  char mine[COLS][ROWS] = { '0' };
  memset(mine, '0', COLS*ROWS*sizeof(mine[0][0])); //利用memset函数把雷盘每个位置初始化为字符‘0'

  char show[COLS][ROWS] = { '0' };
  memset(show, '*', COLS*ROWS*sizeof(show[0][0]));  //把显示盘的每个位置初始化为字符‘*'
  Display(show,COLS,ROWS);

  printf("\n");

  set_mine(mine, COLS, ROWS);     //布雷,把MIX颗雷随机分布在雷盘中
  Display(mine, COLS, ROWS);

  while ( win<( (COLS-2)*(ROWS-2)-MAX ) )       //开始玩游戏的循环
  {
    printf("请输入排雷坐标:");
    scanf_s("%d%d", &x, &y);
    if ((x >= 1) && (x <= COLS-2) && (y >= 1) && (y <= ROWS-2))
    {
      if (show[x][y] == '*')
      {
        if (mine[x][y] == '1')
        {
          if (0 == win) //如果第一次就踩到雷,为了玩家的游戏体验,把这个雷移走
          {
            reset_mine(mine,COLS,ROWS,x,y);  
            int count = get_mine_count(mine, x, y);
            if (count != 0)      //该位置有雷,那就不进行扩展
            {
              show[x][y] = count + '0';
              win++;
              system("cls");
              Display(show, COLS, ROWS);
            }
            else
            {
              show[x][y] = '0';
              win++;
              extend(mine, x, y, &win, show);
              system("cls");
              Display(show, COLS, ROWS);
            }
          }
          else
          {
            printf("你炸了,游戏结束!\n"); //非第一步踩到雷,游戏结束
            Display(mine, COLS, ROWS);
            break;
          }  
        }
        else
        {
          int count = get_mine_count(mine, x, y);
          if (count != 0)      //该位置有雷,那就不进行扩展
          {
            show[x][y] = count + '0';
            win++;
            system("cls");
            Display(show, COLS, ROWS);
          }
          else
          {
            show[x][y] = '0';
            win++;
            extend(mine, x, y, &win, show);
            system("cls");
            Display(show, COLS, ROWS);
          }      
        }
      }
      else
      {
        printf("这个位置已排查,请重新选择坐标\n");
      }
    }
    else
    {
      printf("坐标不合法,请重新输入\n");
    }
  }
  if (win == ((COLS - 2)*(ROWS - 2)) - MAX)
  {
    printf("排雷成功,游戏结束\n");
    Display(mine, COLS, ROWS);
  }
}

void menu()
{
  printf("*******************************\n");
  printf("***** 1. play   2. exit*****\n");
  printf("*******************************\n");
}

void test()
{
  do
  {
    menu();
    int input = 0;
    printf("请选择:");
    scanf_s("%d", &input);

    switch (input)
    {
    case 1:
      game();
      break;
    case 2:
      exit(0);
      break;
    default:
      printf("输入错误\n");
      break;
    }
  } while (1);
}

int main()
{
  test();
}

扫雷成功

扫雷失败

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

相关文章

  • C++ 封装 DLL 供 C# 调用详细介绍

    C++ 封装 DLL 供 C# 调用详细介绍

    这篇文章主要介绍了C++ 封装 DLL 供 C# 调用(以C# 调用C++ 二次封装的VLC播放库为介质,支持回调函数的封装),需要的朋友可以参考下面我文章的具体内容
    2021-09-09
  • VC对自定义资源加密解密(AES)的详解

    VC对自定义资源加密解密(AES)的详解

    本篇文章是对VC对自定义资源加密解密(AES)进行了详细的分析介绍,需要的朋友参考下
    2013-06-06
  • C语言详解如何实现带头双向循环链表

    C语言详解如何实现带头双向循环链表

    带头双向循环链表:结构最复杂,一般用在单独存储数据。实际中使用的链表数据结构,都是带头双向循环链表。另外这个结构虽然结构复杂,但是使用代码实现以后会发现结构会带来很多优势,实现反而简单
    2022-04-04
  • C++ 两个vector对象拼接方式

    C++ 两个vector对象拼接方式

    这篇文章主要介绍了C++ 两个vector对象拼接方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-11-11
  • C语言实现猜数字小游戏

    C语言实现猜数字小游戏

    这篇文章主要为大家详细介绍了C语言实现猜数字小游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-11-11
  • C++实现多项式相乘

    C++实现多项式相乘

    这篇文章主要介绍了C++实现多项式相乘方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-07-07
  • vscode配置远程开发环境并远程调试运行C++代码的教程

    vscode配置远程开发环境并远程调试运行C++代码的教程

    这篇文章主要介绍了vscode配置远程开发环境并远程调试运行C++代码的教程,本文通过截图实例相结合给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-04-04
  • C++实现多线程查找文件实例

    C++实现多线程查找文件实例

    这篇文章主要介绍了C++实现多线程查找文件实例,对于深入学习C++程序设计有着很好的参考借鉴价值,需要的朋友可以参考下
    2014-10-10
  • C++中声明类的class与声明结构体的struct关键字详解

    C++中声明类的class与声明结构体的struct关键字详解

    这篇文章主要介绍了C++中声明类的class与声明结构体的struct关键字,默认情况下结构的所有成员均是公有的,而类的所有成员是私有的,需要的朋友可以参考下
    2016-01-01
  • 解析如何利用switch语句进行字符统计

    解析如何利用switch语句进行字符统计

    本篇文章是对如何利用switch语句进行字符统计的方法进行了详细的分析介绍,需要的朋友参考下
    2013-06-06

最新评论