C++实现控制台版扫雷程序
本文实例为大家分享了C++实现控制台版扫雷程序的具体代码,供大家参考,具体内容如下
测试平台: WIN7
工具: VC6.0 , VS2008都能编译得过。
花了两天时间写的,里面涉及的算法大都是自己想的,所以可能有些BUG。
#include <iostream>
#include <time.h>
#include <windows.h>
using namespace std;
#pragma comment (linker,"/subsystem:console")
#define BLACK 0 //空白
#define MINE 100 //地雷
#define NOSWEEP 0 //未扫过
#define SWEEP 1 //扫雷
#define FLAG 2 //标记
class WinMine
{
public:
WinMine(int iRow, int iColumn);
~WinMine();
public:
void InitMine(int iMineMax);
void SetColor();
void StatisticsMine();
void Map();
bool OpenWhites(int x, int y, int status);
void GameStart(int iMineMax);
void GameOver();
bool GoodOver();
private:
int **m_ppMine;
int **m_ppSweep;
bool m_bMineflag;
bool m_bSweepflag;
int m_row;
int m_column;
int m_minenum;
};
int main(void)
{
system("color 0d");
while (true)
{
int level;
WinMine *Mine;
cout << "请输入游戏等级:" <<endl;
cout << "1.初级" <<endl;
cout << "2.中级" <<endl;
cout << "3.高级" <<endl;
cout << "4.退出" <<endl;
cin >> level;
if (level == 1)
{
Mine = new WinMine(9,9);
Mine->GameStart(10);
}
else if (level == 2)
{
Mine = new WinMine(16,16);
Mine->GameStart(40);
}
else if (level == 3)
{
Mine = new WinMine(16,30);
Mine->GameStart(70);
}
else if (level == 4)
{
return 0;
}
else
{
cout << "输入错误!" <<endl;
continue;
}
delete Mine;
}
return 0;
}
WinMine::WinMine(int iRow, int iColumn)
{
int i;
//储雷状态
m_ppMine = (int **) new int[iRow]; if (!m_ppMine) return;
m_ppMine[0] = new int[iRow * iColumn]; if (!*m_ppMine) { delete[] m_ppMine; m_ppMine = NULL; return;}
m_bMineflag = true;
//扫雷状态
m_ppSweep = (int **) new int[iRow]; if (!m_ppSweep) return;
m_ppSweep[0] = new int[iRow * iColumn]; if (!*m_ppSweep) { delete[] m_ppSweep; m_ppSweep = NULL; return;}
m_bSweepflag = true;
m_row = iRow; m_column = iColumn;
for (i = 1; i < iRow; i++)
{
m_ppMine[i] = m_ppMine[0] + i * iRow;
}
for (i = 1; i < iRow; i++)
{
m_ppSweep[i] = m_ppSweep[0] + i * iRow;
}
memset(m_ppSweep[0], 0, iRow * iColumn * sizeof(int));
memset(m_ppMine[0], 0, iRow * iColumn * sizeof(int));
}
WinMine::~WinMine()
{
if (m_bMineflag)
{
if (m_ppMine[0]) delete[] m_ppMine[0];
if (m_ppMine) delete[] m_ppMine;
}
if (m_bSweepflag)
{
if (m_ppSweep[0]) delete[] m_ppSweep[0];
if (m_ppSweep) delete[] m_ppSweep;
}
}
//设置颜色
void WinMine::SetColor()
{
system("color 0a");
}
//初始化雷数
void WinMine::InitMine(int iMineMax)
{
int x, y,num = 0;
srand( (unsigned)time(NULL) );
for (int i = 0; num != iMineMax; i++)
{
x = rand()%m_row;
y = rand()%m_column;
if (MINE != m_ppMine[x][y])
{
m_ppMine[x][y] = MINE;
num++;
}
}
m_minenum = num;
}
//统计雷数
void WinMine::StatisticsMine()
{
int i, j;
int n, m;
int num = 0; //保存雷数
//中间
for ( i = 1; i < m_row - 1; i++)
{
for ( j = 1; j < m_column - 1; j++)
{
if ( m_ppMine[i][j] == BLACK)
{
for (n = i - 1; n <= i + 1; n++)
{
for (m = j - 1; m <= j + 1; m++)
{
if ( m_ppMine[n][m] == MINE )
num++;
}
}
m_ppMine[i][j] = num;
num = 0;
}
}
}
//顶边
for ( i = 1; i < m_column - 1; i++)
{
if (m_ppMine[0][i] == BLACK)
{
for (n = 0; n < 2; n++)
{
for (m = i - 1; m <= i + 1; m++)
{
if (m_ppMine[n][m] == MINE)
num++;
}
}
m_ppMine[0][i] = num;
num = 0;
}
}
//下边
for ( i = 1; i < m_column - 1; i++)
{
if (m_ppMine[m_row - 1][i] == BLACK)
{
for (n = m_row - 2; n < m_row; n++)
{
for (m = i - 1; m <= i + 1; m++)
{
if (m_ppMine[n][m] == MINE)
num++;
}
}
m_ppMine[m_row - 1][i] = num;
num = 0;
}
}
//左边
for ( i = 1; i < m_row - 1; i++ )
{
if (m_ppMine[i][0] == BLACK)
{
for (n = i - 1; n <= i + 1; n++)
{
for (m = 0; m < 2; m++)
if (m_ppMine[n][m] == MINE)
num++;
}
m_ppMine[i][0] = num;
num = 0;
}
}
//右边
for ( i = 1; i < m_row - 1; i++ )
{
if (m_ppMine[i][m_column - 1] == BLACK)
{
for (n = i - 1; n <= i + 1; n++)
{
for (m = m_column - 2; m < m_column; m++)
{
if (m_ppMine[n][m] == MINE)
num++;
}
}
m_ppMine[i][m_column - 1] = num;
num = 0;
}
}
//左上角
if (m_ppMine[0][0] != MINE)
{
if (m_ppMine[0][1] == MINE)
num++;
if (m_ppMine[1][1] == MINE)
num++;
if (m_ppMine[1][0] == MINE)
num++;
m_ppMine[0][0] = num;
num = 0;
}
//左下角
if (m_ppMine[m_row - 1][0] != MINE)
{
if (m_ppMine[m_row - 2][0] == MINE)
num++;
if (m_ppMine[m_row - 2][1] == MINE)
num++;
if (m_ppMine[m_row - 1][1] == MINE)
num++;
m_ppMine[m_row - 1][0] = num;
num = 0;
}
//右上角
if (m_ppMine[0][m_column - 1] != MINE)
{
if (m_ppMine[1][m_column - 1] == MINE)
num++;
if (m_ppMine[1][m_column - 2] == MINE)
num++;
if (m_ppMine[0][m_column - 2] == MINE)
num++;
m_ppMine[0][m_column - 1] = num;
num = 0;
}
//右下角
if (m_ppMine[m_row - 1][m_column - 1] != MINE)
{
if (m_ppMine[m_row - 2][m_column - 1] == MINE)
num++;
if (m_ppMine[m_row - 2][m_column - 2] == MINE)
num++;
if (m_ppMine[m_row - 1][m_column - 2] == MINE)
num++;
m_ppMine[m_row - 1][m_column - 1] = num;
num = 0;
}
}
//展开空白
bool WinMine::OpenWhites(int row, int column, int status)
{
if (row < 0 || row > (m_row - 1) || column < 0 || column > (m_column - 1))
return true;
if (status == SWEEP && m_ppMine[row][column] == MINE)
{
return false;
}
if (status == FLAG)
{
m_ppSweep[row][column] = FLAG;
return true;
}
if (m_ppSweep[row][column] == NOSWEEP && m_ppMine[row][column] != MINE)
{
if (m_ppMine[row][column] > 0)
{
m_ppSweep[row][column] = SWEEP;
return true;
}
else
{
m_ppSweep[row][column] = SWEEP;
OpenWhites(row-1, column, status);
OpenWhites(row-1, column-1, status);
OpenWhites(row-1, column+1, status);
OpenWhites(row, column-1, status);
OpenWhites(row, column+1, status);
OpenWhites(row+1, column, status);
OpenWhites(row+1, column-1, status);
OpenWhites(row+1, column+1, status);
}
}
return true;
}
//地图
void WinMine::Map()
{
SetColor();
int i, j;
for ( i = 0; i < m_row; i++)
{
for (j = 0; j < m_column; j++)
{
if (m_ppSweep[i][j] == SWEEP)
{
if (m_ppMine[i][j] == 0)
{
cout << "□";
}
else if (m_ppMine[i][j] == 1)
{
cout << "①";
}
else if (m_ppMine[i][j] == 2)
{
cout << "②";
}
else if (m_ppMine[i][j] == 3)
{
cout << "③";
}
else if (m_ppMine[i][j] == 4)
{
cout << "④";
}
else if (m_ppMine[i][j] == 5)
{
cout << "⑤";
}
else if (m_ppMine[i][j] == 6)
{
cout << "⑥";
}
else if (m_ppMine[i][j] == 7)
{
cout << "⑦";
}
else if (m_ppMine[i][j] == 8)
{
cout << "⑧";
}
}
else if (m_ppSweep[i][j] == FLAG)
{
cout << "⊙";
}
else
cout << "▇";
}
cout << endl;
}
}
//游戏结束
void WinMine::GameOver()
{
int i, j;
for ( i = 0; i < m_row; i++)
{
for (j = 0; j < m_column; j++)
{
if (m_ppMine[i][j] == MINE)
cout << "★";
else
cout << "□";
}
cout << endl;
}
}
//检查是否雷标记正确。
bool WinMine::GoodOver()
{
int i, j;
int num = 0;
for (i = 0; i < m_row; i++)
{
for (j = 0; j < m_column; j++)
{
if (m_ppSweep[i][j] == FLAG && m_ppMine[i][j] == MINE)
num++;
}
}
if (num == m_minenum)
return true;
else
return false;
}
//开始游戏
void WinMine::GameStart(int iMineMax)
{
int x, y;
int flag;
begin:
memset(m_ppSweep[0], 0, m_row * m_column * sizeof(int));
memset(m_ppMine[0], 0, m_row * m_column * sizeof(int));
InitMine(iMineMax);
StatisticsMine();
while (true)
{
system("cls");
Map();
cout << "请输入要扫的区域的坐标:" <<endl;
cout << "请输入纵坐标:";cin>>x;
cout << "请输入横坐标:";cin>>y;
if (x <= 0 || x > m_row || y <= 0 || y > m_column)
{
cout <<"输入错误请重新输入" <<endl;
Sleep(1000);
continue;
}
cout << "请输入是要做标记还是扫雷,扫雷请按1,标记请按2:" <<endl;
cin >> flag;
if ( false == OpenWhites(x-1, y-1, flag) )
{
int i;
system("cls");
GameOver();
cout << "游戏结束!" <<endl;
cout << "继续游戏请按1,退出游戏请按0:"<<endl;
cin >> i;
if (i == 1)
goto begin;
else
goto end;
}
else
{
if (GoodOver() == true)
{
int i;
cout << "扫雷成功!" <<endl;
cout << "继续游戏请按1,退出游戏请按0:"<<endl;
cin >> i;
if (i == 1)
goto begin;
else
goto end;
}
}
}
end:
return;
}
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。
赞 (0)
