C语言实现自定义扫雷游戏(递归版)

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

实现过程

对于用C语言实现扫雷游戏得实现,可将游戏过程分为两个板块。

  • 实现游戏关键功能得函数
  • 搭建合理得游戏过程

实现游戏关键功能

为了将游戏功能方便管理和键入,首先我们创建一个头文件,mine.h对游戏功能进行声明。
然后创建对应的源文件mine.c对这些函数进行定义。

对于游戏功能,我们首先想到的是构建一个目标规格的雷盘,也就是二维数组。
为了使游戏更具可玩性,所以雷盘的规格应可以自定义。所以在mine.h头文件中,应先用define定义ROW(行)和列(列)的标识符便于自定义。

#define ROW 15  //先将他们的规格定义为15*15
#define COL 15
#define ROWS ROW+2
#define COLS COL+2

为什么要定义后面两个宏?

对于雷盘的的核心功能实现,必然是能够当玩家选择坐标以后(若该位置不是雷)能够精确的统计以该坐标为核心周围九宫格雷的个数,那么处于边界的地方该如何精准的判断呢?

所以将真正的雷盘数组规格定义ROW+2,COL+2大小,在ROW,COL规格的雷盘中随机置雷,使得游戏结果准确。

  • 定义雷盘和显示盘的初始化

定义两同规格的二维数组,前者是真正的雷区,用‘1’和‘0’来表示雷和无雷,后者是显示区,用于玩家操作和结果显示。

void init(char arr[ROWS][COLS], int rows, int cols, char type) //memset批量操作时传入的是指针,即数组开始地址
{
    memset(arr, type, rows*cols * sizeof(mine[0][0]));
}
  • 在显示区随机放置雷
void setmine(char mine[ROWS][COLS], int count)
{
    srand((unsigned)time(NULL));
    int i = 0;
    int j = 0;
    while (count)
    {
        i = (rand() % ROW) + 1;  //让随机值0-9变成1-10 实现操作区随机放雷
        j = (rand() % COL) + 1;
        if (mine[i][j] == '0')
        {
            mine[i][j] = '1';
            count--;
        }
    }
}
  • 打印界面
void display(char board[ROWS][COLS])
{//从第一行一列开始打因元素
    int i = 0;
    int j = 0;

    printf("\n      |");
    for (i = 1; i <= COL; i++)//打印操作区的列号
    {
        printf("%4d", i);  //打印序号
    }
    printf("\n");
    printf("------|------------------------------------------\n");
    for (i = 1; i <= ROW; i++)
    {
        printf("%4d  |", i);//打印序号
        for (j = 1; j <= COL; j++)
        {
            printf("%4c", board[i][j]);
        }
        printf("\n      \n");
    }
}
  • 统计目标点周围雷的个数
int cal_mine(char mine[ROWS][COLS], int x, int y)
{
    return (mine[x - 1][y - 1] + mine[x - 1][y]
        + mine[x - 1][y + 1] + mine[x][y - 1]
        + mine[x][y + 1] + mine[x + 1][y - 1]
        + mine[x + 1][y] + mine[x + 1][y + 1] - (7 * '0'));//这里减7而不是8。因为这样直接可以表示出数字字符
    //此处的目的是统计周围雷的个数,又因为是ASCII码,'1'和'0'只相差1,然后进去'0'的ASCII值得到1的个数

}

如果目标点周围无雷向外扩展

在扫雷游戏过程中必然会出来一点一大片凹陷的强烈快感,但自己实现的过程中这是如何做到的呢?
原则是如果该点周围无雷,那么将用空格替代,然后向四周伸展将周围的“周围无雷点”继续用空格替代,若四周存在“周围有雷点”,那么停止向周围延伸。

在苦思冥想后,想到了用递归的方法来实现这个一点一大片功能。

void expand(char mine[ROWS][COLS], char board[ROWS][COLS], int x, int y)
{
    if (x >= 1 && x <= ROW&&y >= 1 && y <= COL) // 递归防止越界
    {
        /*if (mine[x][y] == '0')*/
        {
            char count = cal_mine(mine, x, y);
            if (count == '0')
            {
                board[x][y] = ' ';
                if (board[x - 1][y - 1] == '*')
                {
                    expand(mine, board, x - 1, y - 1);
                }
                if (board[x - 1][y] == '*')
                {
                    expand(mine, board, x - 1, y);
                }
                if (board[x - 1][y + 1] == '*')
                {
                    expand(mine, board, x - 1, y + 1);
                }
                if (board[x][y - 1] == '*')
                {
                    expand(mine, board, x, y - 1);
                }
                if (board[x][y + 1] == '*')
                {
                    expand(mine, board, x, y + 1);
                }
                if (board[x + 1][y - 1] == '*')
                {
                    expand(mine, board, x + 1, y - 1);
                }
                if (board[x + 1][y] == '*')
                {
                    expand(mine, board, x + 1, y);
                }
                if (board[x + 1][y + 1] == '*')
                {
                    expand(mine, board, x + 1, y + 1);
                }
            }
            else
            {
                board[x][y] = count;
            }
        }
    }
}
  • 判断胜利

若显示区最终剩下的*等于雷数,那么玩家胜利

int is_win(char board[ROWS][COLS])
{
int count = 0;
int i = 0, j = 0;
for (i = 1; i <= ROW; i++)
{
    for (j = 1; j <= COL; j++)
    {
        if ('*' == board[i][j])
        {
            count++;
        }
    }
}
return count;
}
  • 玩家操作
int player(char mine[ROWS][COLS], char board[ROWS][COLS], int count)
{
    int x = 0, y = 0;
    while (1)//检测输入错误区
    {
        while (2 != scanf("%d %d", &x, &y))
        {
            while ((getchar() != '\n'))  //如果玩家输入了很多非法字符,那么将他们全部吸收
            {
                NULL;
            }
            printf("输的是啥玩意?\n");
        }
        if (x < 1 || x > ROW || y < 1 || y > COL)
        {

            printf("输入越界\n");
        }
        else if (board[x][y] != '*')
        {
            printf("你踩过了!\n");
        }
        else
        {
            if (count == 1 && (mine[x][y] == '1')) //!mine[x][y]=='1'这个条件特别重要!!如果玩家没有踩到雷那么将会使这个位置变成雷!!
        //虽然这样并不影响程序完整运行,因为检验雷点是通过存在的*来计数的,这个值附上去以后肯定恒为1占去了*,但是严谨的玩家肯定说你有BUG
                //所以我们还是尽量写通俗易懂的逻辑程序 下面的while循环会让人感觉是检测,其实是赋值转换
            {
                int i = 0, j = 0;
                while (mine[x][y] == '1')
                {
                    i = (rand() % 10) + 1;
                    j = (rand() % 10) + 1;
                    mine[x][y] = mine[i][j];//将值为符号零的元素赋给操作数,
                }
                mine[i][j] = '1';//赋值如果成功那么便让他成为1
            }
        }
        break;
    }
    if (mine[x][y] == '1')//检验开始
    {
        printf("\n新时代的董存瑞!\n");
        return 1;

    }
    else
    {
        expand(mine, board, x, y);
    }
    return is_win(board);
}

最后将这些声明和定义转入到mine.hmine.c

搭建合理的游戏过程

将这些函数功能合理的整个在一起后,在mian函数存在的源文件中创建定义game()函数,将每一步连接在一起。创建menu()函数显示菜单。

game()

void game()
{
    printf("欢\n"); Sleep(250);
    printf("迎\n"); Sleep(250);
    printf("来\n"); Sleep(250);
    printf("到\n"); Sleep(250);
    printf("caco\n"); Sleep(250);
    printf("的\n"); Sleep(250);
    printf("扫\n"); Sleep(250);
    printf("雷!\n"); Sleep(250);
    printf("村"); Sleep(250);
    printf("!"); Sleep(150);
    printf("!"); Sleep(150);
    char mine[ROWS][COLS] = { 0 };
    char board[ROWS][COLS] = { 0 };
    init(mine, ROWS, COLS, '0');
    init(board, ROWS, COLS, '*');
    int num = 0;
    printf("\n要几个雷\n?");
    scanf("%d", &num);
    setmine(mine, num);
    display(mine);
    int count = 1;
    int ret = 0;

    while (1)
    {
        display(board);
        ret = player(mine, board, count);

        count++;
        if (ret == num)  //和玩家输入得雷数要保持一致
        {
            printf("炸不死得诺贝尔~!\n");
            break;
        }
        if (ret == 1)
        {
            break;   //表示直接菜到雷退出
        }
    }

}

munu()

void menu()
{
    printf("*******************************************\n");
    printf("------ 我可以大口吞下炸弹而不伤身体--------\n");  //label 1. original test.
    printf("------1.  进村   -------------------------\n-------------------------\n");
    printf("------0.  撤退! -------------------------\n");
    printf("*******************************************\n");

}
int main()
{
    int a = 0;
    menu();

    do
    {

        scanf("%d", &a);
        switch (a)
        {
        case 1:
            game();
            menu();
            break;

        case 0:
            return 0;
        default:
            printf("\n进村还是不进?\n");
        }
    } while (a);

}

main()函数

int main()
{
    int a = 0;
    menu();

    do
    {
        scanf("%d", &a);
        switch (a)
        {
        case 1:
            game();
            menu();
            break;

        case 0:
            return 0;
        default:
            printf("\n进村还是不进?\n");
        }
    } while (a);

}

程序展示

mine.h

#pragma once
#define ROW 15
#define COL 15
#define ROWS ROW+2
#define COLS COL+2
void init(char mine[ROWS][COLS], int rows, int cols, char type);//初始化数组
void setmine(char mine[ROWS][COLS], int count);//放置随机雷
void display(char board[ROWS][COLS]);//显示界面
int cal_mine(char mine[ROWS][COLS], int x, int y);//统计周围雷的个数
int player(char mine[ROWS][COLS], char board[ROWS][COLS],int count);//玩家操作
void expand(char mine[ROWS][COLS], char board[ROWS][COLS], int x, int y);//递归展开
int is_win(char board[ROWS][COLS]);//判断胜利

mine.c

前文已展示定义

main.h

#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
#include<Windows.h>
#include"mine.h"
    void game()
    {
        printf("欢\n"); Sleep(250);
        printf("迎\n"); Sleep(250);
        printf("来\n"); Sleep(250);
        printf("到\n"); Sleep(250);
        printf("caco\n"); Sleep(250);
        printf("的\n"); Sleep(250);
        printf("扫\n"); Sleep(250);
        printf("雷!\n"); Sleep(250);
        printf("村"); Sleep(250);
        printf("!"); Sleep(150);
        printf("!"); Sleep(150);
        char mine[ROWS][COLS] = { 0 };
        char board[ROWS][COLS] = { 0 };
        init(mine, ROWS, COLS, '0');
        init(board, ROWS, COLS, '*');
        int num = 0;
        printf("\n要几个雷\n?");
        scanf("%d", &num);
        setmine(mine, num);
        display(mine);
        int count = 1;
        int ret = 0;

        while (1)
        {
            display(board);
            ret = player(mine, board, count);

            count++;
            if (ret == num)  //和玩家输入得雷数要保持一致
            {
                printf("炸不死得诺贝尔~!\n");
                break;
            }
            if (ret == 1)
            {
                break;   //表示直接菜到雷退出
            }
        }

    }
    void menu()
{
    printf("*******************************************\n");
    printf("------ 我可以大口吞下炸弹而不伤身体--------\n");  //label 1. original test.
    printf("------1.  进村   -------------------------\n-------------------------\n");
    printf("------0.  撤退! -------------------------\n");
    printf("*******************************************\n");

}
int main()
{
    int a = 0;
    menu();

    do
    {
        scanf("%d", &a);
        switch (a)
        {
        case 1:
            game();
            menu();
            break;

        case 0:
            return 0;
        default:
            printf("\n进村还是不进?\n");
        }
    } while (a);

}

程序运行

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

(0)

相关推荐

  • C语言实现扫雷程序

    使用C语言实现简单的扫雷程序,主要是对二维数组的运用,我们需要一个头文件,两个源文件来实现. game.h //包含函数的声明,宏定义 test.c //包含主函数,函数调用 game.c //包含函数的定义 整体思路 1.要完成一个简单的扫雷程序,我们需要创建两个二维数组,一个保存我们随机生成的雷,另外一个向外界展示. //使用宏定义定义常量,方便之后对数组的使用 #define ROW 11 //雷 #define COL 11 #define ROWS 9 //棋盘 #define COL

  • C语言实现简单扫雷小程序

    扫雷是一款大众类的益智小游戏.游戏目标是在最短的时间内根据点击格子出现的数字找出所有非雷格子,同时避免踩雷,踩到一个雷即全盘皆输. 首先带大家简单来了解一下扫雷的游戏机制: 随便点一个格子,会有一块,上面的数字就代表数字周围八个格子会有几个雷,是1他周围就只有1个雷,2就有两个雷,以此类推,直到找出全部没有雷的格子就算玩家赢.只要点到一个有雷的格子就判定玩家输. 了解了 机制之后就来编写程序.将程序分为test.c.game.c和game.h三个部分.其中test.c中编写主函数,game.c中

  • C语言代码实现扫雷游戏

    本文实例为大家分享了C语言实现扫雷游戏的具体代码,供大家参考,具体内容如下 该游戏我们也是利用数组完成,设置俩个数组一个mine数组,一个show数组. mine数组存放雷,show数组显示棋盘并且显示该位置是否有雷或者显示该位置周围有几个雷. 数组大小有讲究,我们宏定义变量 ROW COL 为9 定义ROWS COLS为11,我们显示的是9X9的棋盘,也是将雷设置在9X9的位置内,但是我们设置数组是设置11X11,因为这样方便我们遍历9X9棋盘四边位置上某位置四周雷的数目,不然的话会发生越界错

  • C语言实现扫雷小游戏(适合初学者)

    扫雷小游戏作为初学c语言的小白有很大的帮助,其中用到了函数的定义,函数的声明,循环语句,数组思想等等知识,对于代码理解和设计代码思路有很大的帮助,本文就详细介绍了代码的各个步骤和运行结果.希望给到和我一样的小白一点帮助. //game.h头文件 #include<stdio.h> #include<stdlib.h> #include<time.h> #define ROW 9//定义了界面的长宽,可以直更改ROW,COL的值,避免了程序中数字重复出现 #define

  • C语言实现简单扫雷小游戏

    本文实例为大家分享了C语言实现扫雷小游戏的具体代码,供大家参考,具体内容如下 #define _CRT_SECURE_NO_WARNINGS #include <stdio.h> #include <windows.h> #include <time.h> /* 用 C 语言写一个简单的扫雷游戏 */ // 1.写一个游戏菜单 Menu() // 2.开始游戏 // 1.初始化二维数组 Init_Interface() // 2.打印游戏界面 Print_Interfa

  • C语言快速实现扫雷小游戏

    本文实例为大家分享了C语言扫雷小游戏的具体实现代码,供大家参考,具体内容如下 一.分析游戏步骤: 具体步骤如图: 二.代码实现: 游戏步骤想好之后,就是用代码把步骤一步一步的实现.具体代码如下: 1.游戏主要实现: game.c #define _CRT_SECURE_NO_WARNINGS 1 #include"game.h" void game() { char mine[ROW + 2][COL + 2]; //记录地雷的数组 char show[ROW + 2][COL + 2

  • C语言代码实现简单扫雷小游戏

    用C语言写一个简单的扫雷,供大家参考,具体内容如下 1.所需要的知识 c语言的基本语法,简单的二维数组,一点简单的递归知识. 2.总体思路 扫雷游戏主要由3个部分组成,埋雷子,扫雷,判断输赢. 扫雷游戏的主体是两个个字符类型的二维数组.一个是mine[][]它的构成是'0'和'1',其中'0'表示无雷,'1'表示有雷.一个是show[][]它的构成是'*'和'数字'.星号表示未开启的地方,数字表示周围的雷数.这里要注意的是:mine和show的实际大小是11x11,但是展示的效果是 9x9.这样

  • C语言开发简易版扫雷小游戏

    前言: 想起来做这个是因为那时候某天知道了原来黑框框里面的光标是可以控制的,而且又经常听人说起这个,就锻炼一下好了. 之前就完成了那1.0的版本,现在想放上来分享却发现有蛮多问题的,而且最重要的是没什么注释[果然那时候太年轻]!现在看了也是被那时候的自己逗笑了,就修改了一些小bug,增加了算是详尽而清楚的注释,嗯,MSDN上面对各种函数的解释很详细的[又锻炼一下英语],顺便让开头和结尾的展示"动"了起来,就当作1.5的版本好了. 这个只是给出了一个实现的思路,其中肯定也有很多不合理的地

  • 基于C语言实现的扫雷游戏代码

    本文详细讲述了基于C语言实现的扫雷游戏代码,代码中备有比较详细的注释,便于读者阅读和理解.希望对学习游戏开发的朋友能有一点借鉴价值. 完整的实例代码如下: /* 模拟扫雷游戏 */ #include <graphics.h> #include <math.h> #include <stdio.h> #include <dos.h> #include <stdlib.h> #include <conio.h> #include <

  • C语言 扫雷程序的实现

    C语言 扫雷程序的实现 这个游戏可以实现根据选择坐标后所显示无雷区范围的不同分为简单 ,普通和困难 游戏程序中,实现了游戏的计时(分钟)功能,实现了扫雷游戏的最基础功能. 定义了一个9*9的棋盘,玩家只要根据提示选择游戏,和难度: 根据提示输入 合法的坐标,如输入不合法则会进行提示: 扫雷结束则会提示:如触雷则游戏自动结束: 整个游戏设计中主要的函数为: 1.初始化地雷棋盘和显示棋盘 2.棋盘打印函数 3.扫雷函数的实现 4.游戏主函数 大家可以在宏定义中改变棋盘大小,雷的个数,以及显示的无雷区

随机推荐