超级详细三子棋实现思路和方法(附代码)C语言版

2024-01-15 16:40

本文主要是介绍超级详细三子棋实现思路和方法(附代码)C语言版,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

步骤:

第一步:主函数部分

        1.创造一个主函数main()

        2.编写游戏菜单和判断用户的选择

 **ROW和COL:

第二步:游戏功能的实现:game()函数介绍和实现

        game函数的介绍:

        game函数的实现(附代码): 

        第一步:创造一个储存下棋数据的变量,并初始化

        第二步:创造打印棋盘

        第四步:开始下棋

        第五步:判断输赢        

总结(可以直接看结果):


步骤:

第一步:主函数部分

        1.创造一个主函数main()

#include<stdio.h>int main()
{return 0;
}

        2.编写游戏菜单和判断用户的选择

        游戏菜单的实现:

#include<stdio.h>
//游戏菜单函数
void menu()//
{printf("*********************************\n");printf("********      1.play      *******\n");printf("********      0.exit      *******\n");printf("*********************************\n");
}
#include<stdio.h>int main()
{menu();return 0;
}

打印结果为:

         判断用户的选择代码块:

int input = 0;//用来储存用户输入的选项数据,结合下方的scanf函数来观看//在这里用do...while循环的好处就是do...while循环是先运行,再判断的    
//因为这个游戏不需要登陆密码什么的,所以就可以先直接运行,再判断
//如果你想设置登录密码,这里建议使用先判断,再运行的语句,比如while循环或者别的循环语句    
do
{menu();printf("请选择:");//提示用户选择游戏模式scanf("%d", &input);//输入函数,使用户来输入选择模式,数据存储在上面定义的input里switch (input)//根据用户输入的选项来决定游戏模式{case 1:game();//游戏的所有重要功能部分——下文有详细介绍和实现方法break;case 0:printf("退出游戏。\n");break;default:printf("选择错误,请重新选择.\n");break;}
} while (input);

        以下是现阶段完整代码:

 **ROW和COL:

在这里同define定义ROW和COL的原因是,方便以后实现多子棋。

#include<stdio.h>
#define ROW 3
#define COL 3int main()
{int input = 0;do{menu();printf("请选择:");scanf("%d", &input);switch (input)case 1:game();//游戏的所有重要功能部分——下文有详细介绍和实现方法break;case 0:printf("退出游戏。\n");break;default:printf("选择错误,请重新选择.\n");break;}} while (input);return 0;
}

第二步:游戏功能的实现:game()函数介绍和实现

        game函数的介绍:

首先来介绍game()函数的作用:

 game函数里包含了实现整个游戏功能的所有函数,同时它将会单独放置在另一个名为game.c文件里,这样如果以后你想要更新游戏功能,就会方便许多。

接下来我们来对game函数进行实现:

        上文已经说明game()函数里放置的是实现整个游戏功能的所有函数。

        game函数的实现(附代码): 

                第一步:创造一个储存下棋数据的变量,并初始化

void game()
{char board[ROW][COL];//第一步是创造一个棋盘InitBoard(board, ROW, COL);//初始化
}

其中Initboard()为初始化棋盘的函数:

void InitBoard(char board[ROW][COL], int row, int col)
{int i = 0, j = 0;for (i = 0; i < row; i++){for (j = 0; j < col; j++){board[i][j] = ' ';//把棋盘初始化,这样就会呈现一个完整的棋盘//本质上是把二维数组的内容全部初始化为— 空格 —}}
}

        第二步:创造打印棋盘

当然初始化后要打印给玩家看:

void game()
{char board[ROW][COL];//第一步是创造一个棋盘InitBoard(board, ROW, COL);//初始化DisplayBoard(board, ROW, COL);//打印棋盘,本质上是打印数组内容
}

其中Display()为打印棋盘的函数:

void DisplayBoard(char board[ROW][COL], int row, int col)
{int i = 0, j = 0;for (i = 0; i < row; i++){for (j = 0; j < col; j++){if (j < col - 1){printf(" %c |",board[i][j]);}else{printf(" %c ",board[i][j]);}}printf("\n");if (i < row - 1){for (j = 0; j < col; j++){if (j < col - 1){printf("---|");}else{printf("---");}}}printf("\n");}
}

打印结果如下: ,此时方格里的内容就是“空格”了。如果你想把内容初始化为别的,如‘ * 也可以试试:

 这样我们的第一步创造棋盘和打印棋盘就实现了。

        第四步:开始下棋

在这里我们默认玩家先下棋,所以我们要封装一个函数,为玩家下棋的函数,方便后续多次调用

void game()
{char board[ROW][COL];//第一步是创造一个棋盘InitBoard(board, ROW, COL);//初始化DisplayBoard(board, ROW, COL);//打印棋盘,本质上是打印数组内容PlayerMove(board,ROW,COL);//玩家下棋
}

        其中PlayerMove()为玩家下棋时调用的函数:

void PlayerMove(char board[ROW][COL], int row, int col)
{int x = 0, y = 0;printf("轮到玩家:>\n");while (1){printf("请输入下棋的坐标:>");scanf("%d %d", &x, &y);//判断玩家输入的坐标是否合法if (x >= 1 && x <= row && y >= 1 && y <= col){//下棋//判断坐标是否占用if (board[x - 1][y - 1] == ' ')//x和y减1,是因为数组下标是从0开始的{board[x - 1][y - 1] = '*';//玩家的棋子为‘*’,当然你可以改成你喜欢的符号break;//如果下棋成果就跳出循环,负责直到你下对为止}else{printf("坐标被占用,请重新输入.\n");}}else{printf("坐标输入错误,请重新输入:>\n");}}
}

玩家下完棋,我们需要更新一下棋盘的情况,需要再次调用Display()函数来打印棋盘

void game()
{char board[ROW][COL];//第一步是创造一个棋盘InitBoard(board, ROW, COL);//初始化DisplayBoard(board, ROW, COL);//打印棋盘,本质上是打印数组内容PlayerMove(board,ROW,COL);//玩家下棋DisplayBoard(board, ROW, COL);//玩家下完棋,我们需要打印出棋盘,看看情况
}

玩家下完后接下来就是电脑下棋了,这次我们需要封装一个电脑下棋的函数ComputerMove()

void game()
{char board[ROW][COL];//第一步是创造一个棋盘InitBoard(board, ROW, COL);//初始化DisplayBoard(board, ROW, COL);//打印棋盘,本质上是打印数组内容PlayerMove(board,ROW,COL);//玩家下棋DisplayBoard(board, ROW, COL);//玩家下完棋,我们需要打印出棋盘,看看情况ComputerMove(board, ROW, COL);//电脑下棋
}

让我们来实现ComputerMove()函数:

void ComputerMove(char board[ROW][COL], int row, int col)
{printf("轮到电脑:>\n");while (1)//这里为1,是为了让电脑一直下,知道成功就跳出while循环{int x = rand() % row;int y = rand() % col;if (board[x][y] == ' '){board[x][y] = '#';//电脑的棋子为‘#’,如果怕你喜欢别的符号,也可以换掉break;//如果下棋成功就跳出循环}}
}

        *其中rand是生成随机数,当rand()%row后,它们的范围就是0~(row-1),就不怕数组越界了,同理rand()%col也是一个道理。

        当电脑下完棋,我们需要更新一下棋盘的情况,需要再次调用Display()函数来打印棋盘

void game()
{char board[ROW][COL];//第一步是创造一个棋盘InitBoard(board, ROW, COL);//初始化DisplayBoard(board, ROW, COL);//打印棋盘,本质上是打印数组内容PlayerMove(board,ROW,COL);//玩家下棋DisplayBoard(board, ROW, COL);//玩家下完棋,我们需要打印出棋盘,更新棋盘情况ComputerMove(board, ROW, COL);//电脑下棋DisplayBoard(board, ROW, COL);//电脑下完棋,我们需要打印出棋盘,更新棋盘情况
}

        第五步:判断输赢        

写到这里我们是不是感觉少写了一个人判断双方各自下完棋后是否获胜的功能呐?那么就让我们来用代码实现这个功能吧。在实现这个功能的时,我们应该知道,怎么来判断输赢:

 如上图,只要这几个方向的符号一样,那就算赢了。

所以我们要封装一个判断胜负的函数InWin()

至于为什么这个函数会有返回值?我的答案是,如果这个函数有返回值,并且如果返回的是玩家下的符号那就是玩家赢了,反之就是电脑赢了,这样会变得非常方便易于判断。

        InWin()函数非最终版:

char IsWin(char board[ROW][COL], int row, int col)
{int i = 0, j = 0;//判断三行for (i = 0; i < row; i++){int k = 0;for (j = 0; j < col; j++){if (j < col - 1){if (board[i][j] == board[i][j + 1] && board[i][j] != ' '){k++;}}else{if (board[i][j - 1] == board[i][j]){k++;}}}if (k == row){return board[i][j - 1];}}//判断三列for (i = 0; i < col; i++){int k = 0;for (j = 0; j < col; j++){if (j < col - 1){if (board[j][i] == board[j + 1][i] && board[j][i] != ' '){k++;}}else{if (board[j - 1][i] == board[j][i]){k++;}}}if (k == row){return board[j - 1][i];}}//判断对角线int k = 0;for (i = 0; i < row - 1; i++){if (board[i][i] == board[i + 1][i + 1] && board[i][i] != ' '){k++;}if (k == row-1){return board[i][i];}}k = 0;for (i = 0; i < row - 1; i++){int j = row - i - 1;if (board[i][j] == board[i + 1][j - 1] && board[i][j] != ' '){k++;}if (k == row - 1){return board[i][j];}}//继续return 'C';//如果都没有赢的话,我们可以返回一个别的符号,让这盘棋继续进行下去
}

        写到这你会不会认为这个判断输赢的函数就结束了?那你就考虑不周了,因为除了玩家赢和电脑赢这两种情况,还有第三种情况,就是平局,那怎么判断平局?这时我们还需要再次写一个判断平局的函数InFull()

int InFull(char board[ROW][COL], int row, int col)
{int i = 0, j = 0;for (i = 0; i < row; i++){for (j = 0; j < col; j++){if (board[i][j] == ' '){return 0;}}//棋盘满了返回1//棋盘不满返回0}return 1;
}

如果这个棋盘还有空位置,那就返回0,可以继续下棋,如果没有位置了,就返回1,出现平局。

选择我们只需要把InFull()函数让InWin()调用就可以了

        InWin()函数最终版如下:

char IsWin(char board[ROW][COL], int row, int col)
{int i = 0, j = 0;//判断三行for (i = 0; i < row; i++){int k = 0;for (j = 0; j < col; j++){if (j < col - 1){if (board[i][j] == board[i][j + 1] && board[i][j] != ' '){k++;}}else{if (board[i][j - 1] == board[i][j]){k++;}}}if (k == row){return board[i][j - 1];}}//判断三列for (i = 0; i < col; i++){int k = 0;for (j = 0; j < col; j++){if (j < col - 1){if (board[j][i] == board[j + 1][i] && board[j][i] != ' '){k++;}}else{if (board[j - 1][i] == board[j][i]){k++;}}}if (k == row){return board[j - 1][i];}}//判断对角线int k = 0;for (i = 0; i < row - 1; i++){if (board[i][i] == board[i + 1][i + 1] && board[i][i] != ' '){k++;}if (k == row-1){return board[i][i];}}k = 0;for (i = 0; i < row - 1; i++){int j = row - i - 1;if (board[i][j] == board[i + 1][j - 1] && board[i][j] != ' '){k++;}if (k == row - 1){return board[i][j];}}//判断是否平局(棋盘有没有满)int ret = InFull(board, row, col);	//满了返回1,不满返回0if (ret == 1){return 'Q';}return 'C';//继续
}

此时game()函数如下:

        因为IsWin()函数有返回值,所以我们需要定义一个变量来接收它。

​
void game()
{char ret = '\0';//接受游戏状态char board[ROW][COL];//第一步是创造一个棋盘InitBoard(board, ROW, COL);//初始化DisplayBoard(board, ROW, COL);//打印棋盘,本质上是打印数组内容PlayerMove(board,ROW,COL);//玩家下棋DisplayBoard(board, ROW, COL);//玩家下完棋,我们需要打印出棋盘,更新棋盘情况ret=IsWin(board, ROW, COL);ComputerMove(board, ROW, COL);//电脑下棋DisplayBoard(board, ROW, COL);//电脑下完棋,我们需要打印出棋盘,更新棋盘情况ret=IsWin(board, ROW, COL);
}​

完事后对game()函数里的ret接受的值进行判断,再考虑是否继续:

void game()
{char ret = '\0';//接受游戏状态char board[ROW][COL];//创造一个二维数组,棋盘。InitBoard(board, ROW, COL);//初始化棋盘DisplayBoard(board, ROW, COL);//打印棋盘,本质上是打印数组内容while(1){PlayerMove(board,ROW,COL);//玩家下棋DisplayBoard(board, ROW, COL);//玩家下完棋,我们需要打印出棋盘,更新棋盘情况//判断玩家是否赢得游戏ret=IsWin(board, ROW, COL);if (ret != 'C'){break;}ComputerMove(board, ROW, COL);//电脑下棋DisplayBoard(board, ROW, COL);//电脑下完棋,我们需要打印出棋盘,更新棋盘情况//判断电脑是否赢得游戏ret=IsWin(board, ROW, COL);if (ret != 'C'){break;}}if (ret == '*'){printf("玩家赢了\n");}else if (ret == '#'){printf("电脑获胜。\n");}else{printf("平局\n");}DisplayBoard(board, ROW, COL);//游戏结束后,打印棋盘,观看胜利的情况
} 

总结(可以直接看结果):

在前面我们说过game()函数里放置的是实现整个游戏功能的所有函数,同时它将会单独放置在另一个名为game.c文件里,这样如果以后你想要更新游戏功能,就会方便许多

game.h文件内容

#define ROW 3
#define COL 3
#include<stdio.h>
#include<time.h>
#include<stdlib.h>//初始化棋盘
void InitBoard(char board[ROW][COL], int row, int col);
//打印棋盘
void DisplayBoard(char board[ROW][COL], int row, int col);
//玩家下棋
void PlayerMove(char board[ROW][COL], int row, int col);
//电脑下棋
void ComputerMove(char board[ROW][COL], int row, int col);void game();//玩家赢了— *
//电脑赢了— #
//平局    — Q
//四句    — C//判断对局结果
char IsWin(char board[ROW][COL], int row, int col);

game.c文件内容

#include"game.h"void menu()
{printf("*********************************\n");printf("********      1.play      *******\n");printf("********      0.exit      *******\n");printf("*********************************\n");
}void game()
{char board[ROW][COL];//创造一个二维数组,棋盘。InitBoard(board, ROW, COL);//初始化棋盘DisplayBoard(board, ROW, COL);//打印棋盘,本质上是打印数组内容char ret = '\0';//接受游戏状态while (1){PlayerMove(board, ROW, COL);DisplayBoard(board, ROW, COL);//打印棋盘,本质上是打印数组内容//判断玩家是否赢得游戏ret = IsWin(board, ROW, COL);if (ret != 'C'){break;}ComputerMove(board, ROW, COL);//电脑下DisplayBoard(board, ROW, COL);//判断电脑是否赢得游戏ret = IsWin(board, ROW, COL);if (ret != 'C'){break;}}if (ret == '*'){printf("玩家赢了\n");}else if (ret == '#'){printf("你输了,哈哈\n");}else{printf("平局\n");}DisplayBoard(board, ROW, COL);
}void InitBoard(char board[ROW][COL], int row, int col)
{int i = 0, j = 0;for (i = 0; i < row; i++){for (j = 0; j < col; j++){board[i][j] = ' ';}}
}void DisplayBoard(char board[ROW][COL], int row, int col)
{int i = 0, j = 0;for (i = 0; i < row; i++){for (j = 0; j < col; j++){if (j < col - 1){printf(" %c |",board[i][j]);}else{printf(" %c ",board[i][j]);}}printf("\n");if (i < row - 1){for (j = 0; j < col; j++){if (j < col - 1){printf("---|");}else{printf("---");}}}printf("\n");}
}void PlayerMove(char board[ROW][COL], int row, int col)
{int x = 0, y = 0;printf("轮到玩家:>\n");while (1){printf("请输入下棋的坐标:>");scanf("%d %d", &x, &y);//判断玩家输入的坐标是否合法if (x >= 1 && x <= row && y >= 1 && y <= col){//下棋//判断坐标是否占用if (board[x - 1][y - 1] == ' '){board[x - 1][y - 1] = '*';break;}else{printf("坐标被占用,请重新输入.\n");}}else{printf("坐标输入错误,请重新输入:>\n");}}
}void ComputerMove(char board[ROW][COL], int row, int col)
{printf("轮到电脑:>\n");while (1){int x = rand() % row;int y = rand() % col;if (board[x][y] == ' '){board[x][y] = '#';break;}}
}int InFull(char board[ROW][COL], int row, int col)
{int i = 0, j = 0;for (i = 0; i < row; i++){for (j = 0; j < col; j++){if (board[i][j] == ' '){return 0;}}}return 1;
}char IsWin(char board[ROW][COL], int row, int col)
{int i = 0, j = 0;//判断三行for (i = 0; i < row; i++){int k = 0;for (j = 0; j < col; j++){if (j < col - 1){if (board[i][j] == board[i][j + 1] && board[i][j] != ' '){k++;}}else{if (board[i][j - 1] == board[i][j]){k++;}}}if (k == row){return board[i][j - 1];}}//判断三列for (i = 0; i < col; i++){int k = 0;for (j = 0; j < col; j++){if (j < col - 1){if (board[j][i] == board[j + 1][i] && board[j][i] != ' '){k++;}}else{if (board[j - 1][i] == board[j][i]){k++;}}}if (k == row){return board[j - 1][i];}}//判断对角线int k = 0;for (i = 0; i < row - 1; i++){if (board[i][i] == board[i + 1][i + 1] && board[i][i] != ' '){k++;}if (k == row-1){return board[i][i];}}k = 0;for (i = 0; i < row - 1; i++){int j = row - i - 1;if (board[i][j] == board[i + 1][j - 1] && board[i][j] != ' '){k++;}if (k == row - 1){return board[i][j];}}//判断平局int ret = InFull(board, row, col);//满了返回1//不满返回0if (ret == 1){return 'Q';}//继续return 'C';
}

主文件:

demo.c

#include"game.h"int main()
{int input = 0;srand((unsigned int)time(NULL));do{menu();printf("请选择:>");scanf("%d", &input);switch (input){case 1:game();//printf("三子棋游戏\n");break;case 0:printf("退出游戏。\n");break;default:printf("选择错误,请重新选择.\n");break;}} while (input);return 0;
}

这篇关于超级详细三子棋实现思路和方法(附代码)C语言版的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/609545

相关文章

hdu1043(八数码问题,广搜 + hash(实现状态压缩) )

利用康拓展开将一个排列映射成一个自然数,然后就变成了普通的广搜题。 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#include<stdlib.h>#include<ctype.h>#inclu

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

【Prometheus】PromQL向量匹配实现不同标签的向量数据进行运算

✨✨ 欢迎大家来到景天科技苑✨✨ 🎈🎈 养成好习惯,先赞后看哦~🎈🎈 🏆 作者简介:景天科技苑 🏆《头衔》:大厂架构师,华为云开发者社区专家博主,阿里云开发者社区专家博主,CSDN全栈领域优质创作者,掘金优秀博主,51CTO博客专家等。 🏆《博客》:Python全栈,前后端开发,小程序开发,人工智能,js逆向,App逆向,网络系统安全,数据分析,Django,fastapi

活用c4d官方开发文档查询代码

当你问AI助手比如豆包,如何用python禁止掉xpresso标签时候,它会提示到 这时候要用到两个东西。https://developers.maxon.net/论坛搜索和开发文档 比如这里我就在官方找到正确的id描述 然后我就把参数标签换过来

让树莓派智能语音助手实现定时提醒功能

最初的时候是想直接在rasa 的chatbot上实现,因为rasa本身是带有remindschedule模块的。不过经过一番折腾后,忽然发现,chatbot上实现的定时,语音助手不一定会有响应。因为,我目前语音助手的代码设置了长时间无应答会结束对话,这样一来,chatbot定时提醒的触发就不会被语音助手获悉。那怎么让语音助手也具有定时提醒功能呢? 我最后选择的方法是用threading.Time

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo

C#实战|大乐透选号器[6]:实现实时显示已选择的红蓝球数量

哈喽,你好啊,我是雷工。 关于大乐透选号器在前面已经记录了5篇笔记,这是第6篇; 接下来实现实时显示当前选中红球数量,蓝球数量; 以下为练习笔记。 01 效果演示 当选择和取消选择红球或蓝球时,在对应的位置显示实时已选择的红球、蓝球的数量; 02 标签名称 分别设置Label标签名称为:lblRedCount、lblBlueCount

浅谈主机加固,六种有效的主机加固方法

在数字化时代,数据的价值不言而喻,但随之而来的安全威胁也日益严峻。从勒索病毒到内部泄露,企业的数据安全面临着前所未有的挑战。为了应对这些挑战,一种全新的主机加固解决方案应运而生。 MCK主机加固解决方案,采用先进的安全容器中间件技术,构建起一套内核级的纵深立体防护体系。这一体系突破了传统安全防护的局限,即使在管理员权限被恶意利用的情况下,也能确保服务器的安全稳定运行。 普适主机加固措施:

webm怎么转换成mp4?这几种方法超多人在用!

webm怎么转换成mp4?WebM作为一种新兴的视频编码格式,近年来逐渐进入大众视野,其背后承载着诸多优势,但同时也伴随着不容忽视的局限性,首要挑战在于其兼容性边界,尽管WebM已广泛适应于众多网站与软件平台,但在特定应用环境或老旧设备上,其兼容难题依旧凸显,为用户体验带来不便,再者,WebM格式的非普适性也体现在编辑流程上,由于它并非行业内的通用标准,编辑过程中可能会遭遇格式不兼容的障碍,导致操

poj 1258 Agri-Net(最小生成树模板代码)

感觉用这题来当模板更适合。 题意就是给你邻接矩阵求最小生成树啦。~ prim代码:效率很高。172k...0ms。 #include<stdio.h>#include<algorithm>using namespace std;const int MaxN = 101;const int INF = 0x3f3f3f3f;int g[MaxN][MaxN];int n