利用C语言实现简易版扫雷

 更新时间:2021年02月18日 17:32:15   作者:赵铁蛋  
这篇文章主要为大家详细介绍了C语言实现简易版扫雷,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

我和我的父亲都是扫雷的狂热粉,小时候我常常因为技术不好而被父亲嘲笑,那么今天我要来做一个简易版扫雷,回头也给他玩一玩。

首先我们要构建好雷盘的样子,我们理所当然想到利用二维数组。那么请注意:因为我们每一次随机生成的雷盘不能展示给用户,所以显示盘与雷盘要分开,那么我们在这里要用到两个二维数组。一个是雷盘,用来记录随机生成雷的布局,另一个是显示盘,初始化全为*,让用户来扫雷。

具体功能:

  • 先由电脑随机生成雷的分布。
  • 玩家通过输入坐标来选择点。
  • 玩家选择对应点后,对应点将显示周围雷的个数(以该点为中心的一个九宫格)。
  • 玩家需要根据这个个数,自己推断出有雷的地方,不能输入有雷的对应坐标,只能输入没雷的对应坐标。
  • 当输入坐标对应点有雷时,炸“死”,游戏结束。当剩下未选中的点的个数等于总共布下的雷的个数时,游戏胜利。

实现功能的函数:

1.菜单函数,必不可少,是展示给用户的第一界面,这个比较简陋,后期可以考虑做一些fashion一点的。

void Menu()
{
 printf("##########################\n");
 printf("#####choose 1 to play#####\n");
 printf("#####choose 0 to exit#####\n");
 printf("Please input your select here:>\n");
}

2.随机生成雷函数,那么对应雷的横竖坐标就是需要随机生成的数。如果大家看过我之前的三子棋就知道,在这里关于随机生成用到了函数GetIndex,算是一个小优化。

void SetMine(char mine_board[][COL], int row, int col)
{
 srand((unsigned long)time(NULL));
 int mine_num = MINES;
 while (mine_num)
 {
 int i_index = GetIndex(1,col-2);//生成1~10的随机数
 int j_index = GetIndex(1,col-2);
 if (mine_board[i_index][j_index] == '0')
 {
 mine_board[i_index][j_index] = '1';
 mine_num--;
 }
 }
}

3.GetIndex函数,这里要注意start和end表示的是生成的随机数的范围。之前我们 如果用rand生成0~10的随机数,就直接rand%10,但是如果是5 ~10呢?所以我们用(end-start+1)得到的是0 ~end的范围,再加一个start,就能到到start ~end的范围。

static int GetIndex(int start, int end)//static修饰 代表该函数只能在本文件内部使用
{
 return rand() % (end - start + 1) + start;
}

4.得到一个点周围雷的数量函数,这里要注意,我写的这个函数的返回值是整型,但得到的mine_board的和是字符型,那么要通过8-'0'来转换一下。

int GetMineNum(char mine_board[][COL], int i, int j)
{
 return mine_board[i-1][j-1]+mine_board[i-1][j]+ mine_board[i-1][j+1]+ \
 mine_board[i][j-1]+ mine_board[i][j+1]+ \
 mine_board[i+1][j-1]+mine_board[i+1][j]+ mine_board[i+1][j+1]-8*'0';//整型
}

5.显示雷盘函数

void ShowBoard(char show_board[][COL], int row, int col)
{
 int i = 0;
 int j = 0;
 printf(" ");
 for (i = 1; i <= col-2; i++)
 {
 printf("%d ", i);
 }
 printf("\n");
 for (i = 1; i <= col - 1; i++)
 {
 printf("----");
 }
 printf("\n");
 for (i = 1; i <=row - 2; i++)
 {
 printf("%-2d|", i);
 for (j = 1; j <=col - 2; j++)
 {
 printf(" %c |", show_board[i][j]);
 }
 printf("\n");
 int k = 1;
 for (k = 1; k <= col - 1; k++)
 {
 printf("----");
 }
 printf("\n");
 }
}

6.进行游戏函数,以total作为一个衡量标准。输入坐标,判断是否为雷的位置,是则游戏结束,不是则总的方块数减1,继续输入坐标。

void PlayGame(char show_board[][COL], char mine_board[][COL], int row, int col)
{
 int i = 0;
 int j = 0;
 int total = (ROW - 2)*(COL - 2);
 while (1)
 {
 ShowBoard(show_board, row, col);
 printf("Please enter your pos<x,y>");
 scanf("%d%d", &i, &j);
 if (i >= 1 && i < row - 2 && j >= 1 && j < col - 2)
 {
 if (mine_board[i][j] == '0')
 {
 int num = GetMineNum(mine_board, i, j);
 show_board[i][j] = num + '0';
 total--;
 }
 else
 {
 ShowBoard(mine_board, row, col);
 printf("Game over!you lose!:(\n");
 break;
 }
 }
 else
 {
 printf("Error!Try again!\n");
 continue;
 }
 if (total == MINES)
 {
 printf("You win:)!\n");
 break;
 }
 }
}

7.Game函数,初始化两个盘,主要调用了前面的函数,对函数的一个整合,来贯穿一下整个流程。

void Game()
{
 char show_board[ROW][COL];
 char mine_board[ROW][COL];
 memset(show_board, '*', sizeof(show_board));
 memset(mine_board, '0', sizeof(mine_board));

 SetMine(mine_board, ROW, COL);
 PlayGame(show_board, mine_board, ROW, COL);
}

这里还是采用了文件的形式,分为函数声明,函数定义,主函数。

函数声明 Mine.h

#ifndef _MINE_H_
#define _MINE_H_

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

#define ROW 12
#define COL 12
#define MINES 20
void Menu();
void Game();
int GetIndex(int start, int end);
void SetMine(char mine_board[][COL], int row, int col);
int GetMineNum(char mine_board[][COL], int i, int j);
void ShowBoard(char show_board[][COL], int row, int col);
void PlayGame(char show_board[][COL], char mine_board[][COL], int row, int col);
#endif 

函数定义 Mine.c

#include"Mine.h"

void Menu()
{
 printf("##########################\n");
 printf("#####choose 1 to play#####\n");
 printf("#####choose 0 to exit#####\n");
 printf("Please input your select here:>\n");
}

static int GetIndex(int start, int end)//static修饰 代表该函数只能在本文件内部使用
{
 return rand() % (end - start + 1) + start;
}

//随机生成雷
void SetMine(char mine_board[][COL], int row, int col)
{
 srand((unsigned long)time(NULL));
 int mine_num = MINES;
 while (mine_num)
 {
 int i_index = GetIndex(1,col-2);
 int j_index = GetIndex(1,col-2);
 if (mine_board[i_index][j_index] == '0')
 {
 mine_board[i_index][j_index] = '1';
 mine_num--;
 }
 }
}
//得到一个点周围雷的数量
int GetMineNum(char mine_board[][COL], int i, int j)
{
 return mine_board[i-1][j-1]+mine_board[i-1][j]+ mine_board[i-1][j+1]+ \
 mine_board[i][j-1]+ mine_board[i][j+1]+ \
 mine_board[i+1][j-1]+mine_board[i+1][j]+ mine_board[i+1][j+1]-8*'0';//整型
} 
//显示雷盘
void ShowBoard(char show_board[][COL], int row, int col)
{
 int i = 0;
 int j = 0;
 printf(" ");
 for (i = 1; i <= col-2; i++)
 {
 printf("%d ", i);
 }
 printf("\n");
 for (i = 1; i <= col - 1; i++)
 {
 printf("----");
 }
 printf("\n");
 for (i = 1; i <=row - 2; i++)
 {
 printf("%-2d|", i);
 for (j = 1; j <=col - 2; j++)
 {
 printf(" %c |", show_board[i][j]);
 }
 printf("\n");
 int k = 1;
 for (k = 1; k <= col - 1; k++)
 {
 printf("----");
 }
 printf("\n");
 }
}
void PlayGame(char show_board[][COL], char mine_board[][COL], int row, int col)
{
 int i = 0;
 int j = 0;
 int total = (ROW - 2)*(COL - 2);
 while (1)
 {
 ShowBoard(show_board, row, col);
 printf("Please enter your pos<x,y>");
 scanf("%d%d", &i, &j);
 if (i >= 1 && i < row - 2 && j >= 1 && j < col - 2)
 {
 if (mine_board[i][j] == '0')
 {
 int num = GetMineNum(mine_board, i, j);
 show_board[i][j] = num + '0';
 total--;
 }
 else
 {
 ShowBoard(mine_board, row, col);
 printf("Game over!you lose!:(\n");
 break;
 }
 }
 else
 {
 printf("Error!Try again!\n");
 continue;
 }
 if (total == MINES)
 {
 printf("You win:)!\n");
 break;
 }
 }
}
void Game()
{
 char show_board[ROW][COL];
 char mine_board[ROW][COL];
 memset(show_board, '*', sizeof(show_board));
 memset(mine_board, '0', sizeof(mine_board));

 SetMine(mine_board, ROW, COL);
 PlayGame(show_board, mine_board, ROW, COL);
}

主函数 main.c

#include"Mine.h"

int main()
{
 int select = 0;
 int quit = 0;
 while (!quit)
 {
 Menu();
 scanf("%d", &select);
 switch (select)
 {
 case 1:
 Game();
 break;
 case 0:
 printf("OK,byebye!\n");
 quit = 1;
 break;
 default:
 printf("Error!input again!\n");
 break;
 }
 }
 system("pause");
 return 0;
}

运行展示界面:

这是简易版的扫雷,过过干瘾还是可以,但和电脑里的扫雷还是有一定的差别,后期还要做一些优化。

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

相关文章

  • 详解设计模式中的中介者模式在C++编程中的运用

    详解设计模式中的中介者模式在C++编程中的运用

    这篇文章主要介绍了设计模式中的中介者模式在C++编程中的运用,中介者模式将对象间的通信封装到一个类中,将多对多的通信转化为一对多的通信,降低了系统的复杂性,需要的朋友可以参考下
    2016-03-03
  • C++全排列中递归交换法实例详解

    C++全排列中递归交换法实例详解

    在本篇文章里小编给各位整理的是关于C++全排列中递归交换法实例内容,有兴趣的朋友们可以学习下。
    2020-02-02
  • 适合初学者练习的C语言实现三子棋小游戏

    适合初学者练习的C语言实现三子棋小游戏

    今天这篇文章主要介绍给大家分享一个适合初学者练习的利用C语言写三子棋小游戏,用简单的C语言来实现小时候玩的三子棋游戏,下面是人机对战,当然这个代码的电脑对手是人工智障而不是人工智能 详细内容就请跟小编一起来阅读下面文章内容吧
    2021-10-10
  • C语言栈顺序结构实现代码

    C语言栈顺序结构实现代码

    一个能够自动扩容的顺序结构的栈 ArrStack 实例 (GCC编译),有需要的朋友可以参考一下
    2013-10-10
  • 如何用C++实现双向循环链表

    如何用C++实现双向循环链表

    本篇文章是对用C++实现双向循环链表的方法进行了详细的分析介绍,需要的朋友参考下
    2013-05-05
  • C/C++标准库之转换UTC时间到local本地时间详解

    C/C++标准库之转换UTC时间到local本地时间详解

    最近遇到一个问题:数据库中存放的时间为UTC时间,但是现在要求都出来显示的时间为本地时间,所以就用C++实现了,下面这篇文章主要给大家介绍了关于C/C++标准库之转换UTC时间到local本地时间的方法,还有C++中获取UTC时间精确到微秒的实现代码,需要的朋友可以参考下。
    2017-11-11
  • C++类的特种函数生成机制详解

    C++类的特种函数生成机制详解

    这篇文章主要给大家介绍了关于C++类特种函数的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
    2021-09-09
  • C++验证LeetCode包围区域的DFS方法

    C++验证LeetCode包围区域的DFS方法

    这篇文章主要介绍了C++验证LeetCode包围区域的DFS方法,本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-07-07
  • C++基于boost asio实现sync tcp server通信流程详解

    C++基于boost asio实现sync tcp server通信流程详解

    这篇文章主要介绍了C++基于boost asio实现sync tcp server通信的流程,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-07-07
  • C++多线程互斥锁和条件变量的详解

    C++多线程互斥锁和条件变量的详解

    这篇文章主要为大家详细介绍了C++多线程互斥锁和条件变量,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下,希望能够给你带来帮助
    2022-03-03

最新评论