C語言實現(xiàn)掃雷游戲的方法
相信大家對掃雷一定都不陌生吧!但是你敢信僅僅使用一些C語言的初階基礎(chǔ)知識就能夠制作一個掃雷簡單小游戲?今天,我將帶領(lǐng)大家一起走進(jìn)掃雷,探究其基礎(chǔ)算法奧秘。
一、基本思路
首先,既然是玩游戲,我們應(yīng)該設(shè)計相應(yīng)的菜單。
輸入1后繼續(xù)讓玩家選擇游戲難度:
我們在這里設(shè)計簡單模式為6x6大小的面板,雷埋10個;中等模式為10x10大小的面板,雷埋30個;困難模式為14x14大小的面板,雷埋50個。
為了方便雷的布置與顯示,我們設(shè)計了兩張面板:show_board[ROW][COL]和mine_board[ROW][COL],其中show面板負(fù)責(zé)顯示“空格”,mine面板負(fù)責(zé)記錄周圍雷的個數(shù)。
玩家通過輸入每個格子所對應(yīng)的坐標(biāo)進(jìn)行掃雷操作。
char** show_board = (char**)malloc(sizeof(char*) * (ROW + grade * 4));//二維數(shù)組動態(tài)內(nèi)存分配 ? for (int i = 0; i < ROW + grade * 4; i++) { ?? ?show_board[i] = (char*)malloc(sizeof(char) * (COL + grade * 4)); ?? ?memset(show_board[i], ' ', (COL + grade * 4));//數(shù)組全部賦值為空格 } ? char** mine_board = (char**)malloc(sizeof(char*) * (ROW + grade * 4)); ? for (int i = 0; i < ROW + grade * 4; i++) { ?? ?mine_board[i] = (char*)malloc(sizeof(char) * (COL + grade * 4)); ?? ?memset(mine_board[i], '0', (COL + grade * 4));//數(shù)組全部賦值為‘0' }?
二、代碼實現(xiàn)
Step1.畫圖
想要進(jìn)行掃雷游戲,首先應(yīng)該有掃雷界面的設(shè)計,如下:
我們建立Showboard函數(shù)進(jìn)行面板的基本繪制:
static void Showboard(char **board,int grade)//面板制作 { ?? ?printf(" "); ?? ?for (int i = 1; i <= ROW-2+4*grade; i++) { ?? ??? ?printf(" %3d", i); ?? ?} ?? ?printf("\n"); ? ?? ?for (int i = 1; i <= ROW-2+4*grade; i++){ ?? ??? ?printf(" ? "); ?? ??? ?for (int k = 1; k <= COL-2+4*grade; k++) { ?? ??? ??? ?printf("----"); ?? ??? ?} ?? ??? ?printf("\n"); ?? ??? ?printf("%2d|",i); ?? ??? ?for (int j = 1; j <= COL-2+4*grade; j++){ ?? ??? ??? ?printf(" %c |",board[i][j]); ?? ??? ?} ?? ??? ?printf("\n"); ?? ?} ?? ?printf(" ? "); ?? ?for (int k = 1; k <= COL-2+4*grade; k++) { ?? ??? ?printf("----"); ?? ?} ?? ?printf("\n"); }
成果圖如下:
Step2. 埋雷
埋雷應(yīng)做到一下幾點:
1.展示面板show_board(這里傳參后以形參board替代)賦值為1(BOOM宏定義為1) ;
2.為了提高用戶體驗,在被炸后能夠展示出雷的位置,我們又建立了一個lei_board數(shù)組用來顯示,并在埋雷時將對應(yīng)位置賦值為'*';
3.埋雷要具有隨機(jī)性,所以這里引用rand函數(shù)(主函數(shù)要種隨機(jī)數(shù)種子);
static void SetMines(char **board,char **lei_board,int grade) { ?? ?int i = 1; ?? ?int x, y; ?? ?while (i <= NUM+grade*20) { ?? ??? ?x = rand() % (ROW-2+4*grade) + 1; ?? ??? ?y = rand() % (COL-2+4*grade) + 1; ?? ??? ?if (board[x][y] != BOOM) { ?? ??? ??? ?board[x][y] = BOOM; ?? ??? ??? ?lei_board[x][y] = '*'; ?? ??? ??? ?i++; ?? ??? ?} ?? ?} }
Step3.計算周圍雷的個數(shù)
當(dāng)我們計算周圍雷的個數(shù)時,自然而然地就會想到統(tǒng)計周圍九宮格的雷的個數(shù)。如果要搜索的位置在面板中間,這很簡單,但如果是以下這類情況,比如頂點格或者邊格怎么辦?如何知道該算哪個格?
為了避免這種特殊情況所帶來的問題,我們不如直接在外面再套一圈,對于各類面板(即二維數(shù)組)的建立也直接按照增大的來,這樣就無需討論邊角的特殊性了。
統(tǒng)計函數(shù)直接通過簡單的函數(shù)返回來實現(xiàn):
static int CountMine(char **board,int x,int y) { ?? ?return board[x - 1][y - 1] + board[x - 1][y] + board[x - 1][y + 1] + \ ?? ??? ?board[x][y - 1] + board[x][y + 1] + board[x + 1][y - 1] + \ ?? ??? ?board[x + 1][y] + board[x + 1][y + 1] - 8 * '0'; }
三、完整代碼
#include<stdio.h> #include<string.h> #include<stdlib.h> #include<time.h> #include<windows.h> ? #define BOOM '1' #define ROW 8 #define COL 8 #define NUM 10 ? static void SetMines(char **board,char **lei_board,int grade)//埋雷 { ?? ?int i = 1; ?? ?int x, y; ?? ?while (i <= NUM+grade*20) { ?? ??? ?x = rand() % (ROW-2+4*grade) + 1; ?? ??? ?y = rand() % (COL-2+4*grade) + 1; ?? ??? ?if (board[x][y] != BOOM) { ?? ??? ??? ?board[x][y] = BOOM; ?? ??? ??? ?lei_board[x][y] = '*'; ?? ??? ??? ?i++; ?? ??? ?} ?? ?} } static void Showboard(char **board,int grade)//面板制作 { ?? ?printf(" "); ?? ?for (int i = 1; i <= ROW-2+4*grade; i++) { ?? ??? ?printf(" %3d", i); ?? ?} ?? ?printf("\n"); ? ?? ?for (int i = 1; i <= ROW-2+4*grade; i++){ ?? ??? ?printf(" ? "); ?? ??? ?for (int k = 1; k <= COL-2+4*grade; k++) { ?? ??? ??? ?printf("----"); ?? ??? ?} ?? ??? ?printf("\n"); ?? ??? ?printf("%2d|",i); ?? ??? ?for (int j = 1; j <= COL-2+4*grade; j++){ ?? ??? ??? ?printf(" %c |",board[i][j]); ?? ??? ?} ?? ??? ?printf("\n"); ?? ?} ?? ?printf(" ? "); ?? ?for (int k = 1; k <= COL-2+4*grade; k++) { ?? ??? ?printf("----"); ?? ?} ?? ?printf("\n"); } static int CountMine(char **board,int x,int y)//統(tǒng)計周圍雷的數(shù)目 { ?? ?return board[x - 1][y - 1] + board[x - 1][y] + board[x - 1][y + 1] + \ ?? ??? ?board[x][y - 1] + board[x][y + 1] + board[x + 1][y - 1] + \ ?? ??? ?board[x + 1][y] + board[x + 1][y + 1] - 8 * '0'; } ? void Game(int grade)//游戲函數(shù),grade為游戲難度等級 { ?? ? ?? ?char** show_board = (char**)malloc(sizeof(char*) * (ROW + grade * 4)); ?? ?for (int i = 0; i < ROW + grade * 4; i++) { ?? ??? ?show_board[i] = (char*)malloc(sizeof(char) * (COL + grade * 4)); ?? ??? ?memset(show_board[i], ' ', (COL + grade * 4)); ?? ?} ?? ?char** mine_board = (char**)malloc(sizeof(char*) * (ROW + grade * 4)); ?? ?for (int i = 0; i < ROW + grade * 4; i++) { ?? ??? ?mine_board[i] = (char*)malloc(sizeof(char) * (COL + grade * 4)); ?? ??? ?memset(mine_board[i], '0', (COL + grade * 4)); ?? ?} ?? ?char** lei_board = (char**)malloc(sizeof(char*) * (ROW + grade * 4)); ?? ?for (int i = 0; i < ROW + grade * 4; i++) { ?? ??? ?lei_board[i] = (char*)malloc(sizeof(char) * (COL + grade * 4)); ?? ??? ?memset(lei_board[i], ' ', (COL + grade * 4)); ?? ?} ?? ? ?? ?SetMines(mine_board,lei_board,grade); ?? ?int x = 0, y = 0; ?? ?int total = 0; ?? ?while (1) ?? ?{ ?? ??? ? ?? ??? ?Showboard(show_board, grade); ?? ??? ?printf("請輸入坐標(biāo):"); ?? ??? ?//printf("%d", '*'); ?? ??? ?scanf("%d %d", &x, &y); ?? ??? ?if (!(x > 0 && x <= ROW + 4 * grade - 2 && y > 0 && y <= COL + 4 * grade - 2)) { ?? ??? ??? ?printf("掃雷的位置有問題!\n"); ?? ??? ??? ?continue; ?? ??? ?} ?? ??? ?if (show_board[x][y] != ' ') { ?? ??? ??? ?printf("該位置已經(jīng)掃過了!\n"); ?? ??? ??? ?continue; ?? ??? ?} ?? ??? ?if (mine_board[x][y] == '1') { ?? ??? ??? ?printf("對不起,你被炸死了!\n"); ?? ??? ??? ?Showboard(lei_board, grade); ?? ??? ??? ?break; ?? ??? ?} ?? ??? ?else { ?? ??? ??? ?int count = CountMine(mine_board, x, y); ?? ??? ??? ?show_board[x][y] = count + '0'; ?? ??? ??? ?total++; ?? ??? ??? ?system("cls"); ?? ??? ?} ?? ??? ?if ((ROW - 2) * (COL - 2) - total <= NUM) { ?? ??? ??? ?printf("恭喜你,游戲通過!\n"); ?? ??? ??? ?break; ?? ??? ?} ?? ?} ?? ?for (int i = 0; i < ROW + grade * 4; i++) { ?? ??? ?free(show_board[i]); ?? ?} ?? ?free(show_board); ?? ?for (int i = 0; i < ROW + grade * 4; i++) { ?? ??? ?free(mine_board[i]); ?? ?} ?? ?free(mine_board); ?? ?for (int i = 0; i < ROW + grade * 4; i++) { ?? ??? ?free(lei_board[i]); ?? ?} ?? ?free(lei_board); } ? void Menu() { ?? ?printf("#########################################\n"); ?? ?printf("## ? ? ? ? ? ? ? ? 1.play ? ? ? ? ? ? ?##\n"); ?? ?printf("## ? ? ? ? ? ? ? ? 0.exit ? ? ? ? ? ? ?##\n"); ?? ?printf("#########################################\n"); } ? int main() { ?? ?srand((unsigned long)time(NULL)); ?? ?int quit = 0; ?? ?while (!quit) ?? ?{ ?? ??? ?int select; ?? ??? ?Menu(); ?? ??? ?printf("請輸入游戲選項:"); ?? ??? ?scanf("%d", &select); ?? ??? ?switch (select) ?? ??? ?{ ?? ??? ?case 1: ?? ??? ??? ?printf("#########################################\n"); ?? ??? ??? ?printf("## ? ? ? ? ? ? ? ? 1.easy ? ? ? ? ? ? ?##\n"); ?? ??? ??? ?printf("## ? ? ? ? ? ? ? ? 2.middle ? ? ? ? ? ?##\n"); ?? ??? ??? ?printf("## ? ? ? ? ? ? ? ? 3.difficult ? ? ? ? ##\n"); ?? ??? ??? ?printf("#########################################\n"); ?? ??? ??? ?printf("請輸入游戲難度:"); ?? ??? ??? ?scanf("%d", &select); ?? ??? ??? ?switch (select) ?? ??? ??? ?{ ?? ??? ??? ?case 1: ?? ??? ??? ??? ?Game(0); ?? ??? ??? ??? ?break; ?? ??? ??? ?case 2: ?? ??? ??? ??? ?Game(1); ?? ??? ??? ??? ?break; ?? ??? ??? ?case 3: ?? ??? ??? ??? ?Game(2); ?? ??? ??? ??? ?break; ?? ??? ??? ?} ?? ??? ??? ?break; ?? ??? ?case 0: ?? ??? ??? ?quit = 1; ?? ??? ??? ?printf("感謝使用!"); ?? ??? ??? ?break; ?? ??? ?default: ?? ??? ??? ?printf("輸入有誤,請重新輸入!\n"); ?? ??? ??? ?break; ?? ??? ?} ? ?? ?} ?? ? }
期待大家在這個代碼的基礎(chǔ)上加入更多創(chuàng)新與改進(jìn),寫出用戶體驗感更好,更加智能高端的掃雷程序!
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
C++實現(xiàn)LeetCode(7.翻轉(zhuǎn)整數(shù))
這篇文章主要介紹了C++實現(xiàn)LeetCode(7.翻轉(zhuǎn)整數(shù)),本篇文章通過簡要的案例,講解了該項技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-07-07C語言數(shù)據(jù)結(jié)構(gòu)順序表的進(jìn)階講解
程序中經(jīng)常需要將一組數(shù)據(jù)元素作為整體管理和使用,需要創(chuàng)建這種元素組,用變量記錄它們,傳進(jìn)傳出函數(shù)等。一組數(shù)據(jù)中包含的元素個數(shù)可能發(fā)生變化,順序表則是將元素順序地存放在一塊連續(xù)的存儲區(qū)里,元素間的順序關(guān)系由它們的存儲順序自然表示2022-04-04深入探討:main函數(shù)執(zhí)行完畢后,是否可能會再執(zhí)行一段代碼?
本篇文章是對main函數(shù)執(zhí)行完畢后,是否可能會再執(zhí)行一段代碼,進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-05-05C語言實現(xiàn)打印數(shù)組以及打印注意事項說明
這篇文章主要介紹了C語言實現(xiàn)打印數(shù)組以及打印注意事項說明,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-01-01C/C++ 中堆和棧及靜態(tài)數(shù)據(jù)區(qū)詳解
這篇文章主要介紹了C/C++ 中堆和棧及靜態(tài)數(shù)據(jù)區(qū)詳解的相關(guān)資料,需要的朋友可以參考下2017-04-04C++類和對象深入探索之分文件編寫點和圓的關(guān)系詳解
先前把C++類和對象的封裝講完了,并且留下了一個判斷兩個立方體是否相等的案例,但是那么多知識點,僅僅一個案例是不夠的,所以再來一個分文件編寫點圓關(guān)系的案例;創(chuàng)建圓類和點類,圓類包含點類,算是一個嵌套吧,順便復(fù)習(xí)一下分文件編寫的方法,開整2022-05-05