【源码项目+解析】C语言/C++开发,打造一个小项目扫雷小游戏!
一直说写个几百行的小项目,于是我写了一个控制台的扫雷,没有想到精简完了代码才200行左右,不过考虑到这是我精简过后的,浓缩才是精华嘛,我就发出来大家一起学习啦,看到程序跑起来能玩,感觉还是蛮有成就感的~哈哈
扫雷应该属于一款大众游戏,从我初中使用计算机开始,它就被集成到了windows系统中,虽然他是这么经典,我还是要介绍一下他的玩法,然后再考虑在控制台中怎么实现它。
1、游戏的主界面,是一个一个小方格,在小方格上单击左键,可以翻开小方格看看后面有什么。
2、在这些小方格的背后隐藏着雷,如果不幸点中了雷,那么就GameOver了。
如果点中的不是一个雷,那么就是一块空地,这个时候会出现两种情况:
1)用鼠标点中的空地周围八个点内有雷,那么就显示雷的个数
2)用鼠标点中的空地周围没有雷,这个时候就将周围的空地全部显示出来,遇到该显示数字的空地,就将数字显示出来。(仔细观察你会发现,数字会将空地围起来,这是一句废话,但是也值得想一想这是为什么)
3、在小方格上,点击鼠标的右键,可以将一个空地标记为雷,当然这个功能只是为了方便你记忆你之前确定是雷的地方。(还有左右键都点,和点击右键出现?标记,这里就不谈啦)
4、当空地上剩余的格子数和雷的个数一样多,那么这个时候就应该算是胜利啦。
OK~游戏流程说完了,这个时候该谈谈如何实现了。

1、首先需要一张地图,一般情况下我们都可以用一个二维数组表示一个地图,每一个元素代表着扫雷中的一个小方格。相应元素存储0,那么地图上的这个位置就是空地,相应元素存储1,那么就代表这个位置就一颗雷。
2、在控制台上依照二维数组长度和宽度,打印相应的小方块。
3、然后就用鼠标点击那些小方块,对于控制台来讲,在黑框框的区域中是有坐标的,可以使用一些函数捕获到你点击了屏幕的哪一个坐标。
4、对于控制台来说,打印一个字符,有的字符横向占一个位置比如普通的字母数字,有的字符横向占两个位置比如一些图形字符: ①②③■◆等等,这点在控制台编程的时候要注意。
5、当点击屏幕的时候,获取到点击的坐标后,去二维数组中查看相应的位置是雷还是空地,从而做相应的处理。
1)假如点击到了雷,那么就控制游戏结束
2)假如点击到了空地有两种情况
1.点击的空地周围有雷,那么就将雷的个数显示出来
2.假如点击的空地周围没有雷,那么就使用递归的方法去探测周围的点,探测出与其相连的所有周围有雷的点。
这个是我实现的效果,下面就是代码啦:
// saolei.cpp : 定义控制台应用程序的入口点。
//
#include "stdafx.h"
#include
#include
#include
#define Boom 10
int a[10][10] = {0};
COORD TempPos[100] ={0};
int nSign = 0;
/************************************
函数名 : WriteWchar
函数作用: 在控制台相应的坐标上显示一串字符
返回值 : void
参数 : int x 横坐标
参数 : int y 纵坐标
参数 : char szString[] 要显示的字符串
说明 :
************************************/
void WriteWchar(int x,int y,char szString[])
{
HANDLE hOut= GetStdHandle(STD_OUTPUT_HANDLE);
COORD pos = {x*2,y};
SetConsoleCursorPosition(hOut,pos);
printf("%s",szString);
}
/************************************
函数名 : DrawNumber
函数作用: 在相应的坐标上,根据传入的数字,打印相应的数字字符
返回值 : void
参数 : COORD pos 要打印的位置
参数 : int nNumber 要打印的数字
说明 :
************************************/
void DrawNumber(COORD pos,int nNumber)
{
switch (nNumber)
{
case 1:
WriteWchar(pos.X,pos.Y,"①");
break;
case 2:
WriteWchar(pos.X,pos.Y,"②");
break;
case 3:
WriteWchar(pos.X,pos.Y,"③");
break;
case 4:
WriteWchar(pos.X,pos.Y,"④");
break;
case 5:
WriteWchar(pos.X,pos.Y,"⑤");
break;
case 6:
WriteWchar(pos.X,pos.Y,"⑥");
break;
case 7:
WriteWchar(pos.X,pos.Y,"⑦");
break;
case 8:
WriteWchar(pos.X,pos.Y,"⑧");
break;
default:
break;
}
}
/************************************
函数名 : GetNumber
函数作用: 获取一个点的四周有几颗雷
返回值 : int
参数 : COORD pos 要探测的点的坐标
说明 :
************************************/
int GetNumber(COORD pos)
{
int nCount = 0;
for(int i = pos.X-1;i<=pos.X+1;i++)
for (int j = pos.Y-1;j<=pos.Y+1;j++)
{
if (a[j][i] == Boom)
{
nCount++;
}
}
return nCount;
}
/************************************
函数名 : Drawmap
函数作用: 打印一下地图
返回值 : void
说明 :
************************************/
void Drawmap()
{
for (int i =0;i<10;i++)
{
for (int j =0;j<10;j++)
{
WriteWchar(j,i,"■");
}
}
}
/************************************
函数名 : Init
函数作用: 随机生成10个地雷,然后存到数组中
返回值 : void
说明 :
************************************/
void Init()
{
srand(time(NULL));
for (int i =0;i<10;i++)
{
int Temp_x = rand()%10;
int Temp_y = rand()%10;
//判断这个地方是不是已经生成一个雷了,如果没有,赋值为雷
if (a[Temp_x][Temp_y]!=Boom)
{
a[Temp_x][Temp_y] = Boom;
}
//如果是雷,就相当于本次生成没有发生过。。。。。
else
{
i--;
}
}
Drawmap();
}
/************************************
函数名 : IsClose
函数作用: 判断是不是已经探测过的点,由于使用的8方向递归的探测,这样避免重复
返回值 : bool
参数 : COORD posTemp
说明 :
************************************/
bool IsClose(COORD posTemp)
{
for (int i =0;i
{
if(TempPos[i].X == posTemp.X&&TempPos[i].Y == posTemp.Y)
return true;
}
return false;
}
/************************************
函数名 : IsKongdi
函数作用: 判断一个点是空地,还是雷,如果是空地,需要做其他处理
返回值 : void
参数 : COORD pos
说明 :
************************************/
bool IsKongdi(COORD pos)
{
int nNumber = 0;
//1 如果是雷,就直接返回一个false说明要挂了
if (a[pos.Y][pos.X] == Boom)
{
return false;
}
//2 如果不是雷,那么就做后续处理
else
{
//2.1先判断一下周围有几颗雷
nNumber = GetNumber(pos);
if (nNumber!=0){
//有几颗雷,就打印这个数字
DrawNumber(pos,nNumber);
return true;
}
else
{
//如果没有雷,那就先画空地出来,然后向周围扩散去探测其他点
WriteWchar(pos.X,pos.Y," ");
}
}
//2.2点到了空地,但是周围没有雷的情况的处理,继续去探测周围8个点
for(int i = -1;i<=1;i++)
for (int j = -1;j<=1;j++)
{
COORD posTemp = {pos.X+i,pos.Y+j};
//是不是越界了
if (i==0&&j==0||posTemp.X==-1||posTemp.Y==-1||posTemp.X==10||posTemp.Y==10)
continue;
//这个点是不是已经探测过了
if (IsClose(posTemp))
continue;
//这个点没有探测过,就将其加入到数组中,然后使其在以后的探测中,存入
TempPos[nSign++] =posTemp;
IsKongdi(posTemp);
}
return true;
}
/************************************
函数名 : GetMousePosition
函数作用: 获取鼠标点击的位置,假如没有获取到,就返回(-1,-1)
返回值 : COORD 鼠标点击的坐标
说明 :
************************************/
COORD GetMousePosition()
{
HANDLE hIn = GetStdHandle(STD_INPUT_HANDLE);
INPUT_RECORD stcInput = {0};
DWORD buffer;
COORD pos = {-1,-1};
ReadConsoleInput(hIn,&stcInput,1,&buffer);
if (stcInput.EventType == MOUSE_EVENT)
{
MOUSE_EVENT_RECORD stcMouseEnent = stcInput.Event.MouseEvent;
if (stcMouseEnent.dwButtonState ==FROM_LEFT_1ST_BUTTON_PRESSED )
{
pos = stcMouseEnent.dwMousePosition;
pos.X/=2;
}
}
return pos;
}
int _tmain(int argc, _TCHAR* argv[])
{
Init();
COORD pos;
//开始游戏
while(1)
{
//获取鼠标点击位置
pos = GetMousePosition();
if (pos.X!=-1)
{
//如果鼠标点击的位置被探测过了,就开始下一次循环
if (IsClose(pos))
{
continue;
}
TempPos[nSign++] =pos;
bool bIskongdi = IsKongdi(pos);
//点到雷了,就直接退出游戏了。
if (false ==bIskongdi)
{
system("cls");
WriteWchar(20,10,"you lose");
getchar();
break;
}
//检测是不是赢了,赢的条件就是没有被探测的点的个数和雷的个数相等
if (nSign ==90)
{
system("cls");
WriteWchar(20,10,"you win");
}
}
}
return 0;
}
项目不是很长,但是注释我写的还算明白,估计大家都可以看得懂,希望对于新手们有一定的帮助,谢谢大家的支持!!!
看到这里你是不是又学到了很多新知识呢~
如果你很想学编程,小编推荐我的C语言/C++编程学习基地【点击进入】!
都是学编程小伙伴们,带你入个门还是简简单单啦,一起学习,一起加油~
还有许多学习资料和视频,相信你会喜欢的!
涉及:游戏开发、常用软件开发、编程基础知识、课程设计、黑客等等......
【源码项目+解析】C语言/C++开发,打造一个小项目扫雷小游戏!的更多相关文章
- iOS开发之Masonry框架源码深度解析
Masonry是iOS在控件布局中经常使用的一个轻量级框架,Masonry让NSLayoutConstraint使用起来更为简洁.Masonry简化了NSLayoutConstraint的使用方式,让 ...
- Masonry框架源码深度解析
Masonry是iOS在控件布局中经常使用的一个轻量级框架,Masonry让NSLayoutConstraint使用起来更为简洁.Masonry简化了NSLayoutConstraint的使用方式,让 ...
- 源码下载:74个Android开发开源项目汇总
1. ActionBarSherlock ActionBarSherlock应该算得上是GitHub上最火的Android开源项目了,它是一个独立的库,通过一个API和主题,开发者就可以很方便地使用所 ...
- spring源码深度解析— IOC 之 容器的基本实现
概述 上一篇我们搭建完Spring源码阅读环境,spring源码深度解析—Spring的整体架构和环境搭建 这篇我们开始真正的阅读Spring的源码,分析spring的源码之前我们先来简单回顾下spr ...
- 《淘宝数据库OceanBase SQL编译器部分 源码阅读--解析SQL语法树》
淘宝数据库OceanBase SQL编译器部分 源码阅读--解析SQL语法树 曾经的学渣 2014-06-05 18:38:00 浏览1455 云数据库Oceanbase OceanBase是 ...
- Spring框架之beans源码完全解析
导读:Spring可以说是Java企业开发里最重要的技术.而Spring两大核心IOC(Inversion of Control控制反转)和AOP(Aspect Oriented Programmin ...
- Spring框架之spring-web http源码完全解析
Spring框架之spring-web http源码完全解析 Spring-web是Spring webMVC的基础,由http.remoting.web三部分组成. http:封装了http协议中的 ...
- Spring框架之spring-web web源码完全解析
Spring框架之spring-web web源码完全解析 spring-web是Spring webMVC的基础,由http.remoting.web三部分组成,核心为web模块.http模块封装了 ...
- Spring源码深度解析之Spring MVC
Spring源码深度解析之Spring MVC Spring框架提供了构建Web应用程序的全功能MVC模块.通过策略接口,Spring框架是高度可配置的,而且支持多种视图技术,例如JavaServer ...
- Spring框架之spring-webmvc源码完全解析
Spring框架之spring-webmvc源码完全解析 Spring框架提供了构建Web应用程序的全功能MVC模块.Spring MVC分离了控制器.模型对象.分派器以及处理程序对象的角色,支持多种 ...
随机推荐
- 借助FreeHttp任意篡改Websocket报文(Websocket改包)
前言 作为Web应用中最常见的数据传输协议之一的Websocket,在我们日常工作中也势必会经常使用到,而在调试或测试中我们常常也有直接改变Websocket数据报文以确认其对应用的影响的需求,本文将 ...
- Cassandra Vnodes在Cassandra 2.0-4.0中的演进
Vnodes简短历史 Vnodes又叫Virtual Nodes.是Cassandra在1.2版本里引入的功能,已经在生产环境中使用了近8年了.从2.0版本开始,因为默认配置里num_tokens一般 ...
- 关于跨域策略文件crossdomain.xml文件
下载flexpaper源码修改后做成swf阅读器,要加入待阅读的swf文件,可以在flex里调用js的方法来获取swf文件的路径的方法,在js只专注获取路径就行,等着flex来调用:但这里会遇到一个问 ...
- 给编程小白的java JDK安装教程
对刚开始学习编程的新手朋友们来说,安装开发环境可谓是一件头疼的事了,为了不让刚准备入门java的朋友还没写出自己的Hello World就被安装开发环境劝退,所以本篇文章诞生了. 下载JDK 打开浏览 ...
- java 求水仙花数
package com.yc.bean; public class ShuiXianHua { public static void main(String[] args) { /** * 题目:打印 ...
- 如何将tensorflow1.x代码改写为pytorch代码(以图注意力网络(GAT)为例)
之前讲解了图注意力网络的官方tensorflow版的实现,由于自己更了解pytorch,所以打算将其改写为pytorch版本的. 对于图注意力网络还不了解的可以先去看看tensorflow版本的代码, ...
- 抓取 USB 总线LOG
在实际工作中经常会遇到需要分析USB报文的情形.比如对比不同厂家4G/5G模块.解决实际IP over USB传输效率低下问题. 这时候如果能抓取到 USBMOM 总线的报文将会大有裨益.毕竟所有ho ...
- Linux幽灵漏洞修复
1. 漏洞说明 1.1 漏洞原理 glibc是GNU发布的libc库,即c运行库,在glibc库中的__nss_hostname_digits_dots()函数存在一个缓冲区溢出的漏洞,这个漏洞可以经 ...
- netty之bootstrap
转载自https://blog.csdn.net/zxhoo/article/details/17419229 Netty4学习笔记(2)-- Bootstrap Netty4的代码比我想象的要复杂的 ...
- asp.net core 从 3.1 到 5.0
asp.net core 从 3.1 到 5.0 Intro 就在前几天,微软宣布了 .NET5 发布了 RC1 版本,这也意味着 .NET5 的开发基本稳定了,正式发布之前,不会再新增新的 Feat ...