欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

使用C語言實現五子棋游戲

 更新時間:2022年08月03日 16:48:37   作者:7昂7.  
這篇文章主要為大家詳細介紹了使用C語言實現五子棋游戲,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下

本文實例為大家分享了C語言實現五子棋游戲的具體代碼,供大家參考,具體內容如下

一、實現的目的和意義

1、鞏固和加深對c語言知識的理解
2、學會使用編譯器的各種調試
3、提高解決實際問題的能力

二、實現內容描述

實現簡單的人人對戰(zhàn)五子棋,此設計用的是C語言去實現

三、實現原理

采用二位數組保存棋盤信息,棋盤上面的任何一個位置,里頭可以放三個信息:

1、空
2、用戶1的落子
3、用戶2的落子

下棋就是在二維數組種找對應的空位置,進行落子,落完之后立即就要判斷落子位置是否有五字連珠,從而判斷誰數輸誰贏。每走一次棋會有四種情況:

*用戶1贏
*用戶2贏
*平局(這里說的平局是棋盤被占滿的情況)
*沒有出結果 意思就是時此用戶沒贏 下個用繼續(xù)去下棋

四、實現模塊五子棋實現分為三大模塊

文件名                       作用
five_chress.h       五子棋的函數聲明,頭文件聲明等
five_chress.c       五子棋函數接口的實現
main.c              五子棋函數測試功能

五、模塊代碼分析

1、five_chress.h

#pragma once
#include <stdio.h>
#include <string.h>
#include<Windows.h>
#define ROW 10//數組號行號 按需求調整
#define COL 10//數組列數 按需求調整
#define PLAYER1 1//玩家編號,默認棋盤數據是0,玩家1落子,該位置被改成1
#define PLAYER2 2//玩家編號,默認棋盤數據是0,玩家2落子,該位置被改成2
#define NEXT ? ? ? ?0//游戲繼續(xù)
#define PLAYER1_WIN 1//玩家1贏
#define PLAYER2_WIN 2//玩家2贏
#define DRAW ? ? ? ?3//平局

? ?enum Dir{
?? ?LEFT,
?? ?RIGHT,
?? ?UP,
?? ?DOWN,
?? ?LEFT_UP,
?? ?LEFT_DOWN,
?? ?RIGHT_UP,
?? ?RIGHT_DOWN
? ? };//用枚舉去表示4個方向,上下,左右,左上右下,右上左下,統計用戶當前棋子四個方向的棋數是否為大于等于5
? ? void Menu();//實現菜單
? ? void Game();//構建游戲入口Game()函數 ? ? ?

2、five_chress.c

#include "five_chress.h"

int x = 0;
int y = 0;

void Menu()
{
?? ?printf("############################\n");
?? ?printf("## 1. Play ? ? ? ?0. Exit ##\n");
?? ?printf("############################\n");
?? ?printf("Please Select# ");
}

//按照x,y作為起點,按照特定的方向,求連續(xù)相對的最大格式
int ChessCount(int board[][COL], int row, int col, enum Dir d)
{
?? ?int _x = x - 1; //從1
?? ?int _y = y - 1; //從1

?? ?int count = 0;
?? ?while (1){
?? ??? ?switch (d){
?? ??? ?case LEFT://從當前位置向左移動統計 橫坐標不變 縱坐標變了 所以是y--
?? ??? ??? ?_y--;
?? ??? ??? ?break;
?? ??? ?case RIGHT://從當前位置向右移動統計 橫坐標不變 縱坐標變了 所以是_y++
?? ??? ??? ?_y++;
?? ??? ??? ?break;
?? ??? ?case UP://從當前位置向上移動統計 縱坐標不變 橫坐標變了 所以是_x--
?? ??? ??? ?_x--;
?? ??? ??? ?break;
?? ??? ?case DOWN://從當前位置向下移動統計 縱坐標不變 橫坐標變了 所以是_x++
?? ??? ??? ?_x++;
?? ??? ??? ?break;
?? ??? ?case LEFT_UP://從當前位置向左上移動統計 縱,橫變都變了 所以是_x--,_y--
?? ??? ??? ?_x--, _y--;
?? ??? ??? ?break;
?? ??? ?case LEFT_DOWN://從當前位置左下移動統計 縱,橫坐標都變了 所以是_x++,_y--
?? ??? ??? ?_x++, _y--;
?? ??? ??? ?break;
?? ??? ?case RIGHT_UP://從當前位置右上移動統計 縱,橫坐標都變了 所以是_x--,_y++-
?? ??? ??? ?_x--, _y++;
?? ??? ??? ?break;
?? ??? ?case RIGHT_DOWN://從當前位置右下移動統計 縱,橫坐標都變了 所以是_x++,_y++
?? ??? ??? ?_x++, _y++;
?? ??? ??? ?break;
?? ??? ?default:
?? ??? ??? ?//Do Nothing
?? ??? ??? ?break;
?? ??? ?}
?? ??? ?if (_x < 0 || _x > row - 1 || _y < 0 || _y > col - 1){
?? ??? ??? ?break;/表示越界了就停止
?? ??? ?}
?? ??? ?//合法
?? ??? ?if (board[x - 1][y - 1] == board[_x][_y]){
?? ??? ??? ?count++;//統計從舊位置出發(fā)統計和他一樣的個數
?? ??? ?}
?? ??? ?else{
?? ??? ??? ?break;//如果碰到不一樣的就終止循環(huán)
?? ??? ?}
?? ?}
?? ?return count;//返回統計個數
}

//4種情況
//NEXT:表明要繼續(xù)
//PLAYER1_WIN: 用戶1贏了
//PLAYER2_WIN:用戶2贏了
//DRAW: 平局
int IsOver(int board[][COL], int row, int col)
{
?? ?
?? ?int count1 = ChessCount(board, row, col, LEFT) + ChessCount(board, row, col, RIGHT) + 1; //?
?? ?int count2 = ChessCount(board, row, col, UP) + ChessCount(board, row, col, DOWN) + 1; //?
?? ?int count3 = ChessCount(board, row, col, LEFT_UP) + ChessCount(board, row, col, RIGHT_DOWN) + 1; //?
?? ?int count4 = ChessCount(board, row, col, LEFT_DOWN) + ChessCount(board, row, col, RIGHT_UP) + 1; //?

?? ?if (count1 >= 5 || count2 >= 5 || count3 >= 5 || count4 >= 5){
?? ??? ?//有五子連珠
?? ??? ?//一定有人贏
?? ??? ?//x, y
?? ??? ?if (board[x - 1][y - 1] == PLAYER1){
?? ??? ??? ?return PLAYER1_WIN;
?? ??? ?}
?? ??? ?else{
?? ??? ??? ?return PLAYER2_WIN;
?? ??? ?}
?? ?}

?? ?int i = 0;
?? ?for (; i < row; i++){
?? ??? ?int j = 0;
?? ??? ?for (; j < col; j++){
?? ??? ??? ?if (board[i][j] == 0){
?? ??? ??? ??? ?return NEXT;//當前位置還沒有被填寫棋盤不滿 當前用戶沒有贏返回next ?讓下個用戶繼續(xù)下
?? ??? ??? ?}
?? ??? ?}
?? ?}

?? ?return DRAW;//最后一種情況棋盤滿了 為平局
}

void ShowBoard(int board[][COL], int row, int col)
{
?? ?//printf("\e[1;1H\e[2J");這是linux環(huán)境下用的清屏
?? ?//講數組內容,進行可視化
?? ?system("cls");//vs環(huán)境下用的清屏
?? ?
?? ?int i = 0;
?? ?for (i=1; i <= col; i++){
?? ??? ?printf("%3d", i);
?? ?}
?? ?printf("\n");

?? ?for (i = 0; i < row; i++){
?? ??? ?int j = 0;
?? ??? ?printf("%2d",i + 1 );
?? ??? ?for (; j < col; j++){
?? ??? ??? ?if (board[i][j] == 0){
?? ??? ??? ??? ?printf(" . ");
?? ??? ??? ?}
?? ??? ??? ?else if (board[i][j] == PLAYER1){
?? ??? ??? ??? ?printf(" x ");
?? ??? ??? ?}
?? ??? ??? ?else{
?? ??? ??? ??? ?printf (" y ");
?? ??? ??? ?}
?? ??? ?}
?? ??? ?printf("\n");
?? ?}
}

void PlayerMove(int board[][COL], int row, int col, int who)
{
?? ?while (1){
?? ??? ?printf("Player[%d] Please Enter Your Pos# ", who);
?? ??? ?scanf("%d %d", &x, &y);
?? ??? ?if (x < 1 || x > row || y < 1 || y > col){
?? ??? ??? ?printf("Pos Is Not Right!\n");
?? ??? ??? ?continue;
?? ??? ?}
?? ??? ?else if (board[x - 1][y - 1] != 0){
?? ??? ??? ?printf("Pos Is Occupied!\n");
?? ??? ??? ?continue;
?? ??? ?}
?? ??? ?else{
?? ??? ??? ?//合法性,去重
?? ??? ??? ?board[x - 1][y - 1] = who;
?? ??? ??? ?break;
?? ??? ?}
?? ?}
}

void Game()
{
?? ?int board[ROW][COL];
?? ?memset(board, 0, sizeof(board));
?? ?int result = NEXT;
?? ?do{
?? ??? ?ShowBoard(board, ROW, COL);//顯示棋盤
?? ??? ?PlayerMove(board, ROW, COL, PLAYER1);//Player1先走
?? ??? ?result = IsOver(board, ROW, COL);//判斷游戲是否結束
?? ??? ?if (NEXT != result){
?? ??? ??? ?break;//如果返回值不等于NEXT就跳出循環(huán) 說明當前用戶贏了或者平局 如果=NEXT 下個用戶繼續(xù)下
?? ??? ?}
?? ??? ?ShowBoard(board, ROW, COL);//同上
?? ??? ?PlayerMove(board, ROW, COL, PLAYER2);
?? ??? ?result = IsOver(board, ROW, COL);
?? ??? ?if (NEXT != result){
?? ??? ??? ?break;
?? ??? ?}
?? ?} while (1);
?? ?//p1 win, p2 win, draw
?? ?switch (result){
?? ?case PLAYER1_WIN:
?? ??? ?ShowBoard(board, ROW, COL);
?? ??? ?printf("恭喜用戶1,你已經贏了!\n");
?? ??? ?break;
?? ?case PLAYER2_WIN:
?? ??? ?printf("恭喜用戶2,你已經贏了!\n");
?? ??? ?break;
?? ?case DRAW:
?? ??? ?printf("平局\n");
?? ??? ?break;
?? ?default:
?? ??? ?//do nothing!
?? ??? ?break;
?? ?}
}

分析:

我們要先構建游戲入口void game()函數在里面構建上層本調用框架 例如一開始初始化數組把每個位置都置為0,然后構建調用框架:

1、我們要保存二位數組并可視化 就要構建ShowBoard函數;
2、之后 我們要讓用戶下棋 就要構建PlayerMove()函數
3、第一個用戶下完之后就要立馬判斷當前用戶是否贏,因為落子和判定是強相關的 這時候要構建 IsOver()函數
如果此用戶沒贏 那二個用戶繼續(xù)下。
當我們這些函數邏輯都明確好的時候 就用分而治之思想 ,實現他們的功能。
* 1、ShowBoard()函數里面我用到了一個c語言里面的清屏函數printf(“cls”);引上頭文件window.h,這個的作用是在固定地方刷新視圖,把上次的視圖清理掉 ,顯示當前位置視圖。
后面就打印視圖的內容。用戶一和用戶二可以用字符表示,也可以用圓圈圖案表示這個圖案可以在網上找復制一下就可
*2、 Playermove()函數讓用戶下棋,里面先考慮它的合法性 ,不能越界和重復,再然后用我們在five_chrsss.h文件定義宏來賦值 ,賦完之后再終止循環(huán)。
*3、 Isover()函數里面是讓我們判斷誰輸誰贏 換句話游戲是否結束。游戲結束標志是分為四種情況。用戶一贏,用戶二贏,平局,繼續(xù)。前兩種情況則需要判斷從當前位置出發(fā) 沿四個方向當前用戶棋的個數是否為大于等于5,在判斷一下是哪個用戶的棋 就返回哪個用戶。 而這判斷需要在構建并調用ChressCount()函數去統計當前用戶的個數是否大于等于5,有4個方向,例如上和下為一個方向,左和右為一個方向,左上和右下為一個方向,右上和左下為一個方向,分別各自相加之后在加上1就是當前的棋,如果滿足這個4個方向大于等于5的一個條件,我們就能知道一定有人贏,否則在繼續(xù)判斷周圍是否有空位置,有則返回NEXT讓下個用戶繼續(xù)下,沒有則返回平局*ChressCoun()函數具體分析請看代碼塊

這些函數實現完之后再在Game()函數里面來個Switch循環(huán),來結收宏 去打印誰贏 還是平局。

3、main.c

#include "five_chress.h"

int main()
{
?? ?int quit = 0;
?? ?int select = 0;
?? ?while (!quit){
?? ??? ?Menu();
?? ??? ?scanf("%d", &select);
?? ??? ?switch (select){
?? ??? ?case 1:
?? ??? ??? ?Game();
?? ??? ??? ?break;
?? ??? ?case 0:
?? ??? ??? ?quit = 1;
?? ??? ??? ?printf("ByeBye!\n");
?? ??? ??? ?break;
?? ??? ?default:
?? ??? ??? ?printf("Enter Error, Try Again!\n");
?? ??? ??? ?break;
?? ??? ?}
?? ?}

?? ?return 0;
}

main函數作為測試入口,相比較里面的邏輯是很簡單的,創(chuàng)建一個菜單函數和調用一個Game函數,再增添點其他的內容。

六、實現結果

七、總結

實現五子棋給我最大的體會是應用比理論學習難得多,首先要邏輯清楚,也會涉及到各種實際問題,但是加深了我對知識的理解和運用,也知道了只有多練習 多寫代碼才能更好的提升編程技術。同時我也學到了實現一個復雜的項目時,先去構建一個框架,一個清晰的邏輯。再去逐個擊破每個問題,要學會用分而治之的思想去解決問題。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。

相關文章

  • C語言實現自動存取款機模擬系統

    C語言實現自動存取款機模擬系統

    這篇文章主要為大家詳細介紹了C語言實現自動存取款機模擬系統,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-05-05
  • 適合初學者的C語言轉義字符講解

    適合初學者的C語言轉義字符講解

    轉義字符是很多程序語言、數據格式和通信協議的形式文法的一部分。對于一個給定的字母表,一個轉義字符的目的是開始一個字符序列,使得轉義字符開頭的該字符序列具有不同于該字符序列單獨出現(沒有轉義字符開頭)時的語義。因此轉義字符開頭的字符序列被叫做轉義序列
    2022-04-04
  • 基于malloc與free函數的實現代碼及分析

    基于malloc與free函數的實現代碼及分析

    本篇文章介紹了malloc與free函數的實現代碼及分析。需要的朋友參考下
    2013-05-05
  • c++動態(tài)內存管理與智能指針的相關知識點

    c++動態(tài)內存管理與智能指針的相關知識點

    為了更容易同時也更安全地使用動態(tài)內存,新的標準庫提供了兩種智能指針(smart pointer)類型來管理對象,下面這篇文章主要給大家介紹了關于c++動態(tài)內存管理與智能指針的相關知識點,需要的朋友可以參考下
    2022-03-03
  • 簡要解讀C++的動態(tài)和靜態(tài)關聯以及虛析構函數

    簡要解讀C++的動態(tài)和靜態(tài)關聯以及虛析構函數

    這篇文章主要介紹了簡要解讀C++的動態(tài)和靜態(tài)關聯以及虛析構函數,析構函數在C++編程中平時并不是太常用,需要的朋友可以參考下
    2015-09-09
  • C++任意線程通過hwnd實現將操作發(fā)送到UI線程執(zhí)行

    C++任意線程通過hwnd實現將操作發(fā)送到UI線程執(zhí)行

    做Windows界面開發(fā)時,經常需要在多線程環(huán)境中將操作拋到主線程執(zhí)行,下面我們就來學習一下如何在不需要重新定義消息以及接收消息的情況下實現這一要求,感興趣的可以了解下
    2024-03-03
  • C++實現大數乘法算法代碼

    C++實現大數乘法算法代碼

    這篇文章主要介紹了C++實現大數乘法算法代碼的相關資料,需要的朋友可以參考下
    2015-03-03
  • 用c語言編寫一個通訊錄代碼詳解

    用c語言編寫一個通訊錄代碼詳解

    大家好,本篇文章主要講的是用c語言實現一個通訊錄代碼詳解,感興趣的同學趕快來看一看吧,對你有幫助的話記得收藏一下
    2022-01-01
  • C語言雙向鏈表的原理與使用操作

    C語言雙向鏈表的原理與使用操作

    雙向鏈表也叫雙鏈表,是鏈表的一種,它的每個數據結點中都有兩個指針,分別指向直接后繼和直接前驅。本文主要介紹了C語言算法中雙向鏈表的實現,需要的可以參考一下
    2022-05-05
  • 一篇文章帶你了解C語言文件操作中的幾個函數

    一篇文章帶你了解C語言文件操作中的幾個函數

    這篇文章主要介紹了使用C語言操作文件的基本函數整理,包括創(chuàng)建和打開以及關閉文件的操作方法,需要的朋友可以參考下,希望能夠給你帶來幫助
    2021-09-09

最新評論