用C語言實現三子棋遊戲
本文實例為大傢分享瞭C語言實現三子棋遊戲的具體代碼,供大傢參考,具體內容如下
初始工作,通過#define 定義一個標識符來表示一個常量(棋盤的橫縱坐標)
(因為在代碼的編寫中,會有多處需要用到坐標。那麼,在閱讀代碼時就可能會導致讀者疲於理解當前的值到底是代表什麼,即會對代碼的解讀造成困擾。所以,在該代碼中通過使用宏定義解決瞭這個問題,同時也方便後續更改棋盤的大小。)
#define MAX_ROW 3 #define MAX_COL 3
遊戲創建流程
1. 創建棋盤
創建棋盤(二維數組),並將其初始化(空白,無子狀態)。
//1. 棋盤初始化 void init(char chessBoard[MAX_ROW][MAX_COL]) { for (int row = 0; row < MAX_ROW; row++) { for (int col = 0; col < MAX_COL; col++) { //將這個棋盤(二維數組)每個位置都初始化為空 chessBoard[row][col] = ' '; } } }
2. 打印棋盤
打印棋盤,令玩傢能夠看到此時棋盤的狀態。
//2. 打印棋盤 void printChessBoard(char chessBoard[MAX_ROW][MAX_COL]) { printf("+---+---+---+\n"); for(int row = 0; row < MAX_ROW; row++) { printf("|"); for (int col = 0; col < MAX_COL; col++) { printf(" %c |", chessBoard[row][col]); } printf("\n+---+---+---+\n"); } }
3. 玩傢落子
玩傢將通過輸入坐標(row, col)落子。
註意:在玩傢選擇坐標後需要做合法性檢驗,即判斷玩傢落子的位置是否已經有子,如果有,則玩傢需要重新輸入坐標落子。
//3. 玩傢落子 void playMove(char chessBoard[MAX_ROW][MAX_COL]) { while (1) { int row = 0, col = 0; printf("請玩傢通過輸入坐標落子:"); scanf("%d %d", &row, &col); //檢驗玩傢是否輸入錯誤 if (row < 0 || col < 0 || row >= MAX_ROW || col >= MAX_COL) { printf("輸入錯誤!請重新輸入"); continue; } // 檢驗玩傢輸入的位置 是否已經有子(判斷位置是否為空,不為空即有子) else if (chessBoard[row][col] != ' ') { printf("該位置已經有子,請重新輸入"); continue; } else { //沒問題,落子,標記為'x',並退出當前循環 chessBoard[row][col] = 'x'; break; } } }
4. 判定此時勝負情況
4.1 判斷當前勝負情況,有四種獲勝的方式:
1)橫著3個都是一樣的字符;
2)豎著3個都是一樣的字符;
3)1,3象限對角線的3個位置都是一樣的字符;
4)2,4象限對角線的3個位置都是一樣的字符。
//4. 判定勝負情況,檢查棋盤的所有位置,是否符合勝利的4種情況 //玩傢獲勝,返回x; 電腦獲勝,返回o; 和棋返回p; char chessWin(char chessBoard[MAX_ROW][MAX_COL]) { //判斷橫著的3個位置 for (int row = 0; row < MAX_ROW; row++) { if (chessBoard[row][0] != ' ' && chessBoard[row][0] == chessBoard[row][1] && chessBoard[row][0] == chessBoard[row][2]) { return chessBoard[row][0]; } } //判斷豎著的3個位置 for (int col = 0; col < MAX_COL; col++) { if(chessBoard[0][col] != ' ' && chessBoard[0][col] == chessBoard[1][col] && chessBoard[0][col] == chessBoard[2][col] ){ return chessBoard[0][col]; } } //判斷2,4象限對角線的3個位置 if (chessBoard[0][0] != ' ' && chessBoard[0][0] == chessBoard[1][1] && chessBoard[0][0] == chessBoard[2][2]) { return chessBoard[0][0]; } //判斷1,3象限對角線的3個位置 if (chessBoard[2][0] != ' ' && chessBoard[2][0] == chessBoard[1][1] && chessBoard[2][0] == chessBoard[0][2]) { return chessBoard[2][0]; } if (isFull(chessBoard)) { return 'p'; } return ' '; }
4.2 勝負情況判斷後有4種情況,分別是:
1)玩傢獲勝,返回 ‘x’ ;
2)電腦獲勝,返回 ‘o’ ;
3)和棋,棋盤下滿瞭,但是沒有分出勝負,返回 ‘p’ ;
4)棋盤還有位置是空著的,下棋繼續,返回 ‘ ‘ ;
我們在主函數中定義一個字符變量,用來判斷程序結束返回的是什麼,繼而判斷是玩傢獲勝、電腦獲勝、和棋還是遊戲未結束。
根據字符型變量的值得知遊戲的結果(具體過程如後主函數內容所示)。
4.3 判斷是否和棋的函數為:
//4.3 判斷是否和棋(棋盤滿瞭,但未分勝負),返回1表示棋盤滿瞭,返回0表示棋盤未滿 int isFull(char chessBoard[MAX_ROW][MAX_COL]) { for (int row = 0; row < MAX_ROW; row++) { for (int col = 0; col < MAX_COL; col++) { if (chessBoard[row][col] == ' ') { return 0; } } } return 1; }
5. 電腦落子(電腦隨機落子)
通過 rand() 函數實現電腦的隨機落子。
(取餘運算是很好的截短方法,在這裡,我們通過取餘運算將橫縱坐標的值控制在瞭[0,2]之間。)
//5. 電腦落子(隨機) void computerPlayMove(char chessBoard[MAX_ROW][MAX_COL]) { while (1) { //控制落子的坐標 x,y 在[0,2]之間 int row = rand() % MAX_ROW; int col = rand() % MAX_COL; //判斷電腦想落子的位置是否為空(即是否已經有子),如若已經有子,繼續循環 if (chessBoard[row][col] != ' ') { continue; } //如果電腦想落子的位置為空,則將 ‘o' 賦給該位置,並結束循環 chessBoard[row][col] = 'o'; break; } }
6. 判定此時勝負情況
同過程4。
7. 主函數
註意,我們將 字符型二維數組棋盤 寫在瞭主函數中,而非整個程序的開頭。
原因是:如果寫在開頭,那麼這個變量就是全局變量瞭。
全局變量的生命周期是整個程序,會占用更多的內存,而且如果在程序中間已經使用完畢這個變量,也是不能釋放內存的。
此外,全局變量的作用域是整個工程。如果在哪裡將它改動後,整體都會受到影響,在查問題出在哪裡的時候會很麻煩。
所以,本著盡量不使用全局變量的原則,將這個變量寫在瞭主函數中。當其他函數需要這個參數的時候,再進行傳參使用即可。
int main() { char chessBoard[MAX_ROW][MAX_COL]; char winner = ' '; //1. 棋盤初始化 init(chessBoard); while (1) { //2. 打印棋盤,令玩傢觀察此時棋盤的狀態 printChessBoard(chessBoard); //3. 玩傢落子 playMove(chessBoard); //4. 判斷勝負情況 winner = chessWin(chessBoard); if (winner != ' ') { //已經有非空返回值,遊戲結束 break; } //5. 電腦落子 computerPlayMove(chessBoard); //4. 判斷勝負情況 winner = chessWin(chessBoard); if (winner != ' ') { //已經有非空返回值,遊戲結束 break; } } if (winner == 'x') { printChessBoard(chessBoard); printf("恭喜玩傢獲勝!"); } else if (winner == 'o') { printChessBoard(chessBoard); printf("惜敗,電腦獲勝瞭哦~"); } else { printChessBoard(chessBoard); printf("哎呀,你和電腦打成平手瞭呢~"); } system("pause"); return 0; }
以上就是我在寫三子棋遊戲的思路、代碼以及一些需要註意的點。
以上就是本文的全部內容,希望對大傢的學習有所幫助,也希望大傢多多支持WalkonNet。