记得第一次在Windows 98上玩扫雷时,那种既紧张又兴奋的感觉至今难忘。作为程序员,用C语言亲手实现这个经典游戏,不仅是对青春的致敬,更是锻炼编程思维的绝佳方式。今天我们就从最基础的9x9棋盘开始,用纯C语言打造一个控制台版的扫雷游戏。
扫雷的核心机制其实非常简单:
关键设计点:我们使用两个11x11的二维数组(实际使用中间的9x9区域),一个用于存储地雷分布(mine),一个用于显示给玩家(show)。这种双缓冲设计避免了数据混淆。
好的代码组织是项目成功的基础。我们采用典型的C语言模块化设计:
code复制扫雷项目/
├── game.h // 头文件(声明和常量)
├── game.c // 游戏逻辑实现
└── main.c // 程序入口和UI交互
这种分文件的方式有三大优势:
在game.h中,我们定义关键常量和函数原型:
c复制#define ROW 9 // 实际游戏区域行数
#define COL 9 // 实际游戏区域列数
#define ROWS ROW+2 // 包含边界的总行数
#define COLS COL+2 // 包含边界的总列数
#define EASY_COUNT 10 // 地雷数量
// 初始化棋盘
void InitBoard(char arr[ROWS][COLS], int rows, int cols, char set);
// 打印棋盘
void DisplayBoard(char arr[ROWS][COLS], int row, int col);
// 布置地雷
void SetMine(char arr[ROWS][COLS], int row, int col);
// 排查地雷
void FindMine(char mine[ROWS][COLS], char show[ROWS][COLS], int row, int col);
初始化是游戏准备阶段的关键步骤。我们需要:
c复制void InitBoard(char board[ROWS][COLS], int rows, int cols, char set)
{
for(int i=0; i<rows; i++){
for(int j=0; j<cols; j++){
board[i][j] = set; // 统一填充指定字符
}
}
}
实际调用方式:
c复制InitBoard(mine, ROWS, COLS, '0'); // 地雷图初始化
InitBoard(show, ROWS, COLS, '*'); // 显示图初始化
布雷需要保证随机性和不重复性。我们采用经典的rand()方法:
c复制void SetMine(char board[ROWS][COLS], int row, int col)
{
int count = EASY_COUNT;
while(count > 0){
int x = rand()%row + 1; // 1-9随机行
int y = rand()%col + 1; // 1-9随机列
if(board[x][y] == '0'){ // 确保不重复布雷
board[x][y] = '1';
count--;
}
}
}
重要提示:在使用rand()前,必须在main()中调用srand((unsigned)time(NULL))初始化随机种子,否则每次运行游戏的雷区位置都会相同。
显示棋盘需要考虑用户友好性:
c复制void DisplayBoard(char board[ROWS][COLS], int row, int col)
{
printf(" "); // 对齐列标
for(int i=1; i<=col; i++){
printf("%2d ", i); // 打印列号
}
printf("\n");
for(int i=1; i<=row; i++){
printf("%2d ", i); // 打印行号
for(int j=1; j<=col; j++){
printf(" %c ", board[i][j]); // 显示棋盘内容
}
printf("\n");
}
}
这是游戏最核心的算法,需要处理:
c复制int GetMineCount(char mine[ROWS][COLS], int x, int y)
{
// 计算周围8格的地雷总数
return (mine[x-1][y] + mine[x-1][y-1] + mine[x][y-1] +
mine[x+1][y-1] + mine[x+1][y] + mine[x+1][y+1] +
mine[x][y+1] + mine[x-1][y+1] - 8*'0');
}
void FindMine(char mine[ROWS][COLS], char show[ROWS][COLS], int row, int col)
{
int x, y;
int win = 0; // 已探索安全区域计数
while(win < row*col - EASY_COUNT){
printf("请输入要排查的坐标(格式:行 列): ");
scanf("%d %d", &x, &y);
// 坐标合法性检查
if(x<1 || x>row || y<1 || y>col){
printf("坐标超出范围!\n");
continue;
}
// 检查是否已探索
if(show[x][y] != '*'){
printf("该位置已探索!\n");
continue;
}
// 踩雷判定
if(mine[x][y] == '1'){
printf("很遗憾,你踩到地雷了!\n");
DisplayBoard(mine, ROW, COL); // 显示全部地雷
break;
}
// 安全区域处理
int count = GetMineCount(mine, x, y);
show[x][y] = count + '0'; // 转换为ASCII数字
DisplayBoard(show, ROW, COL);
win++;
}
// 胜利判定
if(win == row*col - EASY_COUNT){
printf("恭喜你,成功排除所有地雷!\n");
DisplayBoard(mine, ROW, COL);
}
}
良好的用户界面从清晰的菜单开始:
c复制void menu()
{
printf("\n========== 扫雷游戏 ==========\n");
printf("1. 开始游戏\n");
printf("0. 退出游戏\n");
printf("==============================\n");
printf("请选择: ");
}
使用do-while循环实现可重复游戏:
c复制int main()
{
int input;
srand((unsigned)time(NULL)); // 初始化随机种子
do {
menu();
scanf("%d", &input);
switch(input){
case 1:
game(); // 启动游戏
break;
case 0:
printf("游戏结束,再见!\n");
break;
default:
printf("无效选择,请重新输入!\n");
}
} while(input != 0);
return 0;
}
c复制#pragma once
#include <stdio.h>
#include <stdlib.h>
#include <time.h>
#define ROW 9
#define COL 9
#define ROWS ROW+2
#define COLS COL+2
#define EASY_COUNT 10
// 函数声明
void InitBoard(char arr[ROWS][COLS], int rows, int cols, char set);
void DisplayBoard(char arr[ROWS][COLS], int row, int col);
void SetMine(char arr[ROWS][COLS], int row, int col);
void FindMine(char mine[ROWS][COLS], char show[ROWS][COLS], int row, int col);
c复制#include "game.h"
void InitBoard(char board[ROWS][COLS], int rows, int cols, char set)
{
for(int i=0; i<rows; i++){
for(int j=0; j<cols; j++){
board[i][j] = set;
}
}
}
void DisplayBoard(char board[ROWS][COLS], int row, int col)
{
printf(" ");
for(int i=1; i<=col; i++){
printf("%2d ", i);
}
printf("\n");
for(int i=1; i<=row; i++){
printf("%2d ", i);
for(int j=1; j<=col; j++){
printf(" %c ", board[i][j]);
}
printf("\n");
}
}
void SetMine(char board[ROWS][COLS], int row, int col)
{
int count = EASY_COUNT;
while(count > 0){
int x = rand()%row + 1;
int y = rand()%col + 1;
if(board[x][y] == '0'){
board[x][y] = '1';
count--;
}
}
}
int GetMineCount(char mine[ROWS][COLS], int x, int y)
{
return (mine[x-1][y] + mine[x-1][y-1] + mine[x][y-1] +
mine[x+1][y-1] + mine[x+1][y] + mine[x+1][y+1] +
mine[x][y+1] + mine[x-1][y+1] - 8*'0');
}
void FindMine(char mine[ROWS][COLS], char show[ROWS][COLS], int row, int col)
{
int x, y;
int win = 0;
while(win < row*col - EASY_COUNT){
printf("请输入要排查的坐标(格式:行 列): ");
scanf("%d %d", &x, &y);
if(x<1 || x>row || y<1 || y>col){
printf("坐标超出范围!\n");
continue;
}
if(show[x][y] != '*'){
printf("该位置已探索!\n");
continue;
}
if(mine[x][y] == '1'){
printf("很遗憾,你踩到地雷了!\n");
DisplayBoard(mine, ROW, COL);
break;
}
int count = GetMineCount(mine, x, y);
show[x][y] = count + '0';
DisplayBoard(show, ROW, COL);
win++;
}
if(win == row*col - EASY_COUNT){
printf("恭喜你,成功排除所有地雷!\n");
DisplayBoard(mine, ROW, COL);
}
}
void game()
{
char mine[ROWS][COLS] = {0};
char show[ROWS][COLS] = {0};
InitBoard(mine, ROWS, COLS, '0');
InitBoard(show, ROWS, COLS, '*');
SetMine(mine, ROW, COL);
DisplayBoard(show, ROW, COL);
FindMine(mine, show, ROW, COL);
}
c复制#include "game.h"
void menu()
{
printf("\n========== 扫雷游戏 ==========\n");
printf("1. 开始游戏\n");
printf("0. 退出游戏\n");
printf("==============================\n");
printf("请选择: ");
}
int main()
{
int input;
srand((unsigned)time(NULL));
do {
menu();
scanf("%d", &input);
switch(input){
case 1:
game();
break;
case 0:
printf("游戏结束,再见!\n");
break;
default:
printf("无效选择,请重新输入!\n");
}
} while(input != 0);
return 0;
}
虽然我们已经实现了一个完整的扫雷游戏,但仍有改进空间:
可以通过宏定义实现不同难度:
c复制// 在game.h中添加
#define MEDIUM_ROW 16
#define MEDIUM_COL 16
#define MEDIUM_COUNT 40
#define HARD_ROW 30
#define HARD_COL 16
#define HARD_COUNT 99
当点击到周围无雷的空白区域时,可以自动展开所有相邻的空白区域,这是标准扫雷的核心特性之一。可以通过递归算法实现。
添加标记疑似地雷位置的功能,增强游戏体验:
c复制// 在FindMine函数中添加标记逻辑
if(show[x][y] == '*'){
show[x][y] = 'F'; // F表示标记
} else if(show[x][y] == 'F'){
show[x][y] = '*'; // 取消标记
}
记录玩家完成游戏所用的时间,增加挑战性:
c复制#include <time.h>
// 在game()函数中
time_t start = time(NULL);
// 游戏结束时
time_t end = time(NULL);
printf("用时: %.0f秒\n", difftime(end, start));
实现这些功能时,建议采用增量开发的方式,一次只添加一个新特性,并充分测试确保不影响原有功能。