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

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

服务器之家 - 编程语言 - C/C++ - C语言实现简易的扫雷小游戏

C语言实现简易的扫雷小游戏

2022-11-24 11:39Monster_ii C/C++

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

这是一个用C语言实现的控制台扫雷小游戏,实现了随机布置炸弹、扫描炸弹、标记炸弹、百分百第一次不被炸死等功能。

编译器:vs2015

功能模块图

C语言实现简易的扫雷小游戏

源代码

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
#include<stdio.h>
#include<stdlib.h>
#include<time.h>
 
void show(int cbd[10][10],int u[10][10])  //界面输出函数
{
    int i, j;
    //for (i = 0; i < 10; i++)  //输出全部炸弹信息(上帝视角,调试用)
    //{
    //  for (j = 0; j < 10; j++)
    //  {
    //      printf("%2d ", cbd[i][j]);
    //  }
    //  printf("\n");
    //}
    printf("     1 2 3 4 5 6 7 8 9 10\n"); //行标
    printf("    --------------------\n");
    for (i = 0; i<10; i++)  //根据u数组和cbd数组的值选择输出
    {            //■:未扫描,□:周围无炸弹,◇:炸弹标记
        for (j = 0; j<10; j++)
        {
            if (j == 0)printf("%2d║ ", i+1);
            if (u[i][j] == 0)printf("■");
            else if (u[i][j] == 1)
            {
                if (cbd[i][j] == 0)printf("□");
                else printf("%2d", cbd[i][j]);
            }
            else if (u[i][j] == -1) 
            {
                printf("◇");
            }
        }
        printf("\n");
    }
}
 
int find(int cbd[10][10], int i, int j)  //扫描周围炸弹数目
{
    int l, c,lmin,cmin, lmax, cmax, num = 0;
    l = lmin = (i - 1) >= 0 ? (i - 1) : 0;  //计算行、列起点
    c = cmin = (j - 1) >= 0 ? (j - 1) : 0;
    lmax = ((i + 1) < 10 ? (i + 1) : 9);  //计算行、列终点
    cmax = ((j + 1) < 10 ? (j + 1) : 9);
 
    for (l = lmin; l <= lmax; l++)
    {
        for (c = cmin; c <= cmax; c++)
        {
            if (l == i&&c == j)continue;  //跳过自身
            if (cbd[l][c] == -1)num++;  //炸弹计数
        }
    }
    return num;
}
 
void init(int cbd[10][10])   //cbd数组初始化(随机布置炸弹并填充各个格子的数目)
{
    int n = 10, l, c, i, j;
    srand(time(NULL));    //随机数种子设置
    for (i = 0; i< 10; i++)
    {
        for (j = 0; j< 10; j++)
        {
            cbd[i][j] = 0;  //全0填充
        }
    }
    while (n--)   //随机行列放置炸弹
    {
        l = rand() % 10;
        c = rand() % 10;
        if (cbd[l][c] != -1)cbd[l][c] = -1;   //炸弹重复处理
        else n++;
    }
    for (i = 0; i< 10; i++)
    {
        for (j = 0; j< 10; j++)
        {
            if (cbd[i][j] != -1)
            {
                cbd[i][j] = find(cbd, i, j); //炸弹数目填充
            }
        }
    }
}
 
void open(int cbd[10][10],int u[10][10], int i, int j)  //周围无炸弹时的展开函数(递归展开)
{
    u[i][j] = 1;
    if (cbd[i][j] == 0)
    {
        if(i - 1 >= 0 && u[i - 1][j] != 1)open(cbd, u, i - 1, j);   //上
        if(j - 1 >= 0 && u[i][j - 1] != 1)open(cbd, u, i, j - 1);   //左
        if (i + 1 <= 9 && u[i + 1][j] != 1)open(cbd, u, i + 1, j);  //下
        if (j + 1 <= 9 && u[i][j + 1] != 1)open(cbd, u, i, j + 1);  //右
    }
}
 
int judge(int u[10][10]) //判断游戏是否通关
{
    int i, j, num = 0;
    for (i = 0; i < 10; i++)
    {
        for (j = 0; j < 10; j++)
        {
            if (u[i][j] == 0 || u[i][j] == -1)num++;
        }
    }
    if (num == 10)return 1;
    else return 0;
}
 
void fail(int cbd[10][10], int u[10][10])  //游戏失败函数
{
    int i, j;
    for (i = 0; i < 10; i++)   //输出全部炸弹信息
    {
        for (j = 0; j < 10; j++)
        {
            if (cbd[i][j] == 0)printf("  ");
            else if (cbd[i][j] != -1)printf("%2d", cbd[i][j]); 
            else printf("●");
        }
        printf("\n");
    }
    printf("请大侠重新来过~\n");
}
 
void menu(int cbd[10][10],int u[10][10])  //菜单函数
{
    int chs,i,j;
    int boom = 10;  //剩余炸弹数
    int num = 0;   //步数
    while (1)                                   
    {
        system("cls");
        show(cbd,u);
        printf("还有%d个炸弹\n",boom);
        printf("1.扫描    2.标记    3.取消标记\n");
        printf("请输入选项:>");
        scanf("%d", &chs);
        if (chs > 3 || chs < 1) {
            printf("请输入正确选项!\n");
            system("pause");
            continue;
        }
        printf("请输入坐标:>");
        scanf("%d %d", &i, &j);
        if (i < 1 || i>10)
        {
            printf("请输入正确坐标!\n");
            system("pause");
            continue;
        }
        i--;  //输入行列数处理(适用数组下标)
        j--;
        if (chs == 1)
        {
            if (cbd[i][j] == -1) {
                if (num == 0) { //若第一步扫描到炸弹,就重新初始化棋盘
                    init(cbd);
                    show(cbd, u);
                    open(cbd, u, i, j);
                    num++;
                    continue;
                }
                else
                {
                    fail(cbd, u);  //否则游戏失败,循环跳出
                    break;
                }
            }
            if (u[i][j] == 1)  //重复扫描时的处理
            {
                printf("该位置已经扫描过了!\n");
                system("pause");
                continue;
            }
            open(cbd, u, i, j);
            num++;
        }
        else if(chs == 2)
        {
            if (u[i][j] == 1) {
                printf("该位置已经扫描过了!\n"); //只能标记未扫描到的
                system("pause");
                continue;
            }
            u[i][j] = -1;
            boom --;            
        }
        else if (chs == 3)
        {
            if (u[i][j] != -1) {
                printf("该位置没有标记!\n"); //只能取消标记过的
                system("pause");
                continue;
            }
            u[i][j] = 0;
            boom++;
        }
        if (judge(u))  //判断游戏是否通关
        {
            printf("Old Fe 666!\n"); //老铁666!,循环跳出
            break;
        }
    }
}
 
int main()
{
    int cbd[10][10] = { 0 }; //保存炸弹信息的二维数组
    int u[10][10] = { 0 };  //保存是否翻开信息的二维数组
    init(cbd);   //数组初始化
    menu(cbd, u);  //进入菜单
    return 0;
}

Tips:在扫描到周围无炸弹的格子时,要将此格子周围的部分依次打开,直到上下左右都出现炸弹数字为止,这里要用到递归的方法,我的顺序为依次递归处理上、左、下、右的格子,这里顺序可以任意。

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

原文链接:https://blog.csdn.net/Monster_ii/article/details/81269978

延伸 · 阅读

精彩推荐
  • C/C++使用C++实现迷宫游戏

    使用C++实现迷宫游戏

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

    Viitasca6532021-08-27
  • C/C++详解C++模板编程中typename用法

    详解C++模板编程中typename用法

    typename在C++类模板或者函数模板中经常使用的关键字,此时作用和class相同,只是定义模板参数,下面通过例子给大家介绍c++模板typename的具体用法,一起看...

    Erice_s11032021-11-22
  • C/C++详解C语言中的常量指针和指针常量

    详解C语言中的常量指针和指针常量

    这篇文章主要介绍了详解C语言中的常量指针和指针常量,包括其之间的区别是C语言入门学习中的基础知识,需要的朋友可以参考下...

    zinss269147772021-03-05
  • C/C++详解C++编程中的条件判断语句if-else与switch的用法

    详解C++编程中的条件判断语句if-else与switch的用法

    这篇文章主要介绍了C++编程中的条件判断语句if-else与switch的用法,是C++入门学习中的基础知识,需要的朋友可以参考下...

    C++教程网6332021-03-22
  • C/C++基于C++类型重定义的使用详解

    基于C++类型重定义的使用详解

    本篇文章是对C++中类型重定义的使用进行了详细的分析介绍,需要的朋友参考下...

    C++教程网4362020-11-30
  • C/C++C++Zip压缩解压缩示例(支持递归压缩)

    C++Zip压缩解压缩示例(支持递归压缩)

    C++Zip压缩解压缩示例,用第三方函数封装而成,支持 UNCODE, ANSCII、支持压缩文件夹、支持递归压缩...

    C++教程网13372021-01-10
  • C/C++Qt+OpenCV实现目标检测详解

    Qt+OpenCV实现目标检测详解

    这篇文章主要介绍了如何利用Qt和OpenCV中自带xml文件实现目标检测,文中的实现过程讲解详细,感兴趣的小伙伴可以动手试一试...

    SongpingWang8062022-10-13
  • C/C++详谈signed 关键字

    详谈signed 关键字

    c++中关键字有几十个,其中类型修饰关键字有long, short, singed, unsigned。今天我们就来谈一下经常被大家忽视的signed关键字...

    C++教程网5802021-02-21