【C语言】三子棋游戏实现代码

目录

1.三子棋代码功能介绍

2.三子棋游戏实现步骤

①打印菜单栏

②判断是否进入三子棋游戏 

③三子棋游戏基本函数实现

(1)清空(初始化)棋盘函数实现

(2)打印棋盘函数实现

(3)玩家下棋函数实现 

(4)电脑下棋函数实现

(5)判断输赢函数

(6)判断平局函数

④整个游戏算法实现(实现逻辑)

3.三子棋游戏完整代码实现

①game.h头文件

②progress.c源文件

③game.c源文件

 4.结语



1.三子棋代码功能介绍

①用坐标代替点位来下棋

②与电脑进行对局

输赢的判断

2.三子棋游戏实现步骤

①打印菜单栏

void menu()//打印三子棋菜单栏
{
	printf("*************************\n");
	printf("*******THREECHESS********\n");
	printf("#####1.PLAY   0.EXIT#####\n");
	printf("*************************\n");

}

②判断是否进入三子棋游戏 

通过scanf函数来获取用户输入;

再使用switch语句来判断是否进入或退出三子棋游戏;

void progress()//判断是否进入游戏
{
	int input = 0;
	do
	{
		menu();
		printf("请输入>:");
		scanf_s("%d", &input);
		switch (input)
		{
		case 1://如果用户输入1就进入游戏
			printf("进入游戏\n");
			game();//三子棋游戏实现函数
			printf("\n\n\n");//打印空行使得排版更好看
			break;
		case 0:
			printf("退出游戏\n");
			break;
		default://输入不为1或0的情况处理
			printf("输入错误\n");
			break;
		}
		
	}
	while (input);//如果input不为0则一直进入循环判断是否进入三子棋
}

如图所示:

③三子棋游戏基本函数实现

注:首先要知道我们用二维字符数组(char board[ROW][COL])来抽象实现三子棋;

        ROW,COL宏定义为3;

        col,rol传给函数的内容也是3;

(1)清空(初始化)棋盘函数实现

初始化棋盘实际就是将二维数组的各个元素全都初始化为空格,用嵌套for循环来实现

void InitBoard(char board[ROW][COL], int row, int col)//初始化棋盘,这里ROW,COL是宏定义值是3
{
	int i = 0;
	int j = 0;
	for (i = 0; i < row; i++)//用for循环来逐一初始化
	{
		for (j = 0; j < col; j++)
		{
			board[i][j] = ' ';//初始化为空格
	    }
	}
}

(2)打印棋盘函数实现

类似于初始化棋盘函数打印棋盘函数也使用嵌套的for循环语句

并借用键盘上的字符‘|’‘-’分割棋盘形状

void DisPlayBoard(char board[ROW][COL], int row, int col)//打印棋盘,ROW,COL宏定义为3,col,
//rol传的内容也是3
{
	int i = 0;
	int j = 0;
	for (i = 0; i < row; i++)
	{
		for (j = 0; j < col; j++)//for循环来逐一打印
		{
			printf(" %c ",board[i][j]);
			if (j < col - 1)
			{
				printf("|");//打印‘|’来分割每个格子
			}	
		}
		if(i < row - 1)
		    printf("\n---|---|---\n");//一行打印完用横线分割下一行,进行下一行的打印
	}
}

运行结果如下:

(3)玩家下棋函数实现 

:玩家下棋用‘*’, 电脑下棋用‘#’

        要用if语句判断玩家输入的坐标是否在3*3棋盘内;

        还要用if语句判断玩家输入的坐标是否被占用(是否已经下过)

void PlayerMove(char board[ROW][COL], int row, int col)//玩家下棋
{
	int x = 0;
	int y = 0;
	printf("\n玩家下棋>>");//提示玩家下棋
	while (1)//如果输入坐标不正确就多次输入
	{
		printf("\n请输入坐标:\n");
		scanf("%d%d", &x, &y);
		if (x > 0 && x <= row && y > 0 && y <= col)//判断坐标是否在3*3棋盘内
		{
			if (board[x - 1][y - 1] == ' ')//判断该格子是否被占用
			{
				board[x - 1][y - 1] = '*';//没有占用就放字符‘*’作为玩家下过的标志
				
				break;//玩家已经下过一次后跳出循环进行下一轮
			}
			else
				printf("该坐标已被占用,请重新输入.\n");//如果坐标被占用进行提示重新输入
		}
		else
			printf("坐标输入非法,请重新输入.\n");//如果坐标非法提示重新输入
	}
}

运行结果如下:

(4)电脑下棋函数实现

:使用随机数生成函数(rand)来获得电脑要下的坐标;

        获得的随机数除以3取余数就可以获得大于等于0小于3的整数以此来作为电脑下棋的坐标;

        同时也要判断随机生成的坐标是否合法

        如果坐标合法且未被占用就标记‘#’;

void ComputerMove(char board[ROW][COL], int row, int col)//电脑下棋
{
	printf("\n电脑下棋>>\n");//提示电脑下棋
	int x = 0;
	int y = 0;
	while (1)//如果坐标非法就重新输入
	{
		x = rand() % row;//使用随机数生成函数生成随机数,并除以3得余数获得0~2的数
		y = rand() % col;
		{
			if (board[x][y] == ' ')//判断该坐标在棋盘上是否被占用
			{
				board[x][y] = '#';//没被占用就标记‘#’
				break;
			}			
		}
	}
}

(5)判断输赢函数

:判断输赢函数多增加了一个参数tmp来传入玩家和电脑代表的字符,一旦一列或一行或从左往右或从右往左的直线有三个相同的字符并且等于tmp的就返回字符tmp,通过tmp就可以判断出玩家赢了还是电脑赢了

char IsWin(char board[ROW][COL], int row, int col, char tmp)//判断输赢,返回字符
{
		int i = 0;
		int j = 0;
		int count = 0;
		for (i = 0; i < ROW; i++)//判断一是否有三个字符相同
		{
			int count = 0;
			for (j = 0; j < COL; j++)
			{
				if (board[i][j] != tmp)//有一个不同就break跳出本次循环
					break;
				else
					count++;//有一个相同count就+1
				if (count == 3)//当count等于3时就代表一行中有三个相同
					return tmp;//当一行中有三个相同的字符时就代表该字符所对应的赢了返回该字符
			}
			
		}
		for (j = 0; j < COL; j++)//同理判断一列中是否有三个字符相同
		{
			int count = 0;

			for (i = 0; i < ROW; i++)
			{
				if (board[i][j] != tmp)
					break;
				else
					count++;
				if (count == 3)
					return tmp;
			}
			
		}
		for (count = 0,i = 0; i < ROW; i++)//判断从左往右的直线三个字符是否相同
		{
			
			if (board[i][i] != tmp)
				break;
			else
				count++;
			if (count == 3)
				return tmp;	
		}
		
			for (count = 0,j = COL - 1,i =0; j >= 0,i <ROW; j--,i++)//判断从左往右的直线
			{
				
				if (board[i][j] != tmp)
					break;
				else
					count++;
				if (count == 3)
					return tmp;
			}
		
		return 0;
}

(6)判断平局函数

:除了判断输赢还要记得判断是否有平局的情况;

        用嵌套for循环来逐一判断二位数组是否有空格;

        没有空格代表平局,返回字符‘Q’

char IsFull(char board[ROW][COL], int row, int col)//判断平局
{
	int i = 0;
	int j = 0;
	for (i = 0; i < row; i++)
	{
		for (j = 0; j < col; j++)//使用for循环逐一判断棋盘是否有空格
		{
			if (board[i][j] == ' ')
				return 0;//有空格也就是还没下完返回0
		}
	}
	return 'Q';//没有空格平局了返回字符‘Q’
}

④整个游戏算法实现(实现逻辑)

void game()//游戏的整个算法实现
{
	char ret = 0;
	char bian = 0;
	char board[ROW][COL] = { 0 };
	InitBoard(board, ROW, COL);//清空棋盘
	DisPlayBoard(board, ROW, COL);//打印棋盘

	while (1)
	{
		PlayerMove(board, ROW, COL);//玩家下棋
		DisPlayBoard(board, ROW, COL);//打印棋盘
		ret = IsWin(board, ROW, COL, '*');//判断输赢
		bian = IsFull(board, ROW, COL);//判断是否平局
		if (ret == '*')
		{
			printf("\n玩家赢啦~");
			break;
		}
		if (bian == 'Q')
		{
			printf("\n平局");
			break;
		}
		ComputerMove(board, ROW, COL);//电脑下棋
		DisPlayBoard(board, ROW, COL);//打印棋盘
		ret = IsWin(board, ROW, COL, '#');//判断输赢
		bian = IsFull(board, ROW, COL);//判断是否平局
		if (ret == '#')
		{
			printf("\n太可惜了,电脑赢了");
			break;
		}
		if (bian == 'Q')
		{
			printf("\n平局");
			break;
		}
	}
}

运行结果如下:

3.三子棋游戏完整代码实现

:有些注释上文标注但在完整代码中未标注

①game.h头文件

#pragma once
#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
#include<string.h>
#include<stdlib.h>
#include<time.h>
constexpr auto ROW = 3;
constexpr auto COL = 3;
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);//电脑下棋函数
char IsWin(char board[ROW][COL], int row, int col, char tmp);//判断输赢函数
char IsFull(char board[ROW][COL], int row, int col);//判断平局函数



②progress.c源文件

#include"game.h"
void menu()//打印菜单栏
{
	
	printf("*************************\n");
	printf("*******THREECHESS********\n");
	printf("#####1.PLAY   0.EXIT#####\n");
	printf("*************************\n");


}
void game()//游戏的整个算法实现
{
	char ret = 0;
	char bian = 0;
	char board[ROW][COL] = { 0 };
	InitBoard(board, ROW, COL);//清空棋盘
	DisPlayBoard(board, ROW, COL);//打印棋盘

	while (1)
	{
		PlayerMove(board, ROW, COL);//玩家下棋
		DisPlayBoard(board, ROW, COL);//打印棋盘
		ret = IsWin(board, ROW, COL, '*');//判断输赢
		bian = IsFull(board, ROW, COL);//判断是否平局
		if (ret == '*')
		{
			printf("\n玩家赢啦~");
			break;
		}
		if (bian == 'Q')
		{
			printf("\n平局");
			break;
		}
		ComputerMove(board, ROW, COL);//电脑下棋
		DisPlayBoard(board, ROW, COL);//打印棋盘
		ret = IsWin(board, ROW, COL, '#');//判断输赢
		bian = IsFull(board, ROW, COL);//判断是否平局
		if (ret == '#')
		{
			printf("\n太可惜了,电脑赢了");
			break;
		}
		if (bian == 'Q')
		{
			printf("\n平局");
			break;
		}
	}
}





void progress()//判断是否进入游戏
{
	int input = 0;
	
	do
	{
		menu();
		printf("请输入>:");
		scanf_s("%d", &input);
		switch (input)
		{
		case 1:
			printf("进入游戏\n");
			game();
			printf("\n\n\n");
			break;
		case 0:
			printf("退出游戏\n");
			break;
		default:
			printf("输入错误\n");
			break;
		}
		
	}
	while (input);

}

int main()	
{
	int row = ROW;
	int col = COL;
	srand((unsigned int)time(NULL));//设置随机数的生成起点
	progress();
	return 0;
	}

③game.c源文件

#include"game.h"
void InitBoard(char board[ROW][COL], int row, int col)//初始化棋盘
{
	int i = 0;
	int 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;
	int j = 0;
	for (i = 0; i < row; i++)
	{
		for (j = 0; j < col; j++)
		{
			printf(" %c ",board[i][j]);
			if (j < col - 1)
			{
				printf("|");
			}	
		}
		if(i < row - 1)
		    printf("\n---|---|---\n");
	}
}

void PlayerMove(char board[ROW][COL], int row, int col)//玩家下棋
{
	int x = 0;
	int y = 0;
	printf("\n玩家下棋>>");
	while (1)
	{
		printf("\n请输入坐标:\n");
		scanf("%d%d", &x, &y);
		if (x > 0 && x <= row && y > 0 && 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电脑下棋>>\n");
	int x = 0;
	int y = 0;
	while (1)
	{
		x = rand() % row;
		y = rand() % col;
		{
			if (board[x][y] == ' ')
			{
				board[x][y] = '#';
				break;
			}			
		}
	}
}



char IsWin(char board[ROW][COL], int row, int col, char tmp)//判断输赢
{
		int i = 0;
		int j = 0;
		int count = 0;
		for (i = 0; i < ROW; i++)
		{
			int count = 0;

			for (j = 0; j < COL; j++)
			{
				if (board[i][j] != tmp)
					break;
				else
					count++;
				if (count == 3)
					return tmp;
			}
			
		}
		for (j = 0; j < COL; j++)
		{
			int count = 0;

			for (i = 0; i < ROW; i++)
			{
				if (board[i][j] != tmp)
					break;
				else
					count++;
				if (count == 3)
					return tmp;
			}
			
		}
		for (count = 0,i = 0; i < ROW; i++)
		{
			
			if (board[i][i] != tmp)
				break;
			else
				count++;
			if (count == 3)
				return tmp;	
		}
		
			for (count = 0,j = COL - 1,i =0; j >= 0,i <ROW; j--,i++)
			{
				
				if (board[i][j] != tmp)
					break;
				else
					count++;
				if (count == 3)
					return tmp;
			}
		
		return 0;
}

char IsFull(char board[ROW][COL], int row, int col)//判断平局
{
	int i = 0;
	int j = 0;
	for (i = 0; i < row; i++)
	{
		for (j = 0; j < col; j++)
		{
			if (board[i][j] == ' ')
				return 0;
		}
	}
	return 'Q';
}



	

 4.结语

以上就是C语言实现简单三子棋游戏完整代码以及过程实现啦,我们也可以通过三子棋的代码思考五子棋游戏的实现,以及对于随机数生成来作为电脑下棋的坐标似乎有点愚笨,我们或许可以试着写一些让电脑选择坐标更聪明一点的函数,让玩家拥有更好的游戏体验哦,希望能和大家一起学习进步。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/375306.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

科研绘图-半小提琴图-

文章目录 前言1.软件安装-Origin 20222.绘制半小提琴图3.绘制径向条形图 前言 本文叙述记录的是一些科研绘图的实现方法&#xff0c;具体介绍从软件安装到实现图表绘制的详细过程。 1.软件安装-Origin 2022 Origin是一款具有丰富绘图功能的科研绘图软件&#xff0c;安装过程…

【Leetcode】1696. 跳跃游戏 VI

文章目录 题目思路代码结果 题目 题目链接 给你一个下标从 0 开始的整数数组 nums 和一个整数 k 。 一开始你在下标 0 处。每一步&#xff0c;你最多可以往前跳 k 步&#xff0c;但你不能跳出数组的边界。也就是说&#xff0c;你可以从下标 i 跳到 [i 1&#xff0c; min(n -…

Kuberntes权威指南

一、目录 二、Kubernetes入门 三、Kubernetes核心原理 四、Kubernetes开发指南 五、Kubernetes运维指南 六、Kubernetes高级案例进阶 七、Kubernetes源码导读

【BIAI】Lecture 12 - Emotion in Brain AI

Emotion in Brain & AI 专业术语 Limbic system 边缘系统 Amygdala 杏仁核 temporal lobe 颞叶 hippocampus 海马体 Central Nucleus 中央核 medial amygdala 内侧杏仁核 ventral periaqueductal gray 腹侧中脑导水管周围灰质 课程大纲 What is emotion 当大脑检测到某些具…

C++适配器——stack queue

栈和队列 本章思维导图&#xff1a; 注&#xff1a;本章思维导图对应的.xmind和.png文件都已同步导入至资源&#xff0c;可免费查看 文章目录 栈和队列1. 适配器2. 栈 stack2.1 概念及结构2.2 使用2.3 模拟实现 3. 队列 queue3.1 普通队列 queue3.1.1 概念及结构3.1.2 使用3.1…

vue3 watch和watchEffect

Watch监听ref定义的数据 1.ref数据基本数据类型 let sumref&#xff08;0&#xff09; const stopWatchwatch&#xff08;sum,(new,old)>{ If(new>10){ stopWatch() } console.log(‘sum数据变化了’) }&#xff09;2.ref数据为对象类型,监听的是对象的地址值,若想监听…

Android:Android Studio安装及环境配置

1开发环境搭建 Android开发需要使用java的jdk环境,所以需要下载JAVA JDK。 1.1安装配置JAVA JDK Java的JDK下载: https://www.oracle.com/technetwork/java/javase/downloads/index.html 配置java的环境变量: JAVA_HOME:java安装路径。 新增环境变量CLASSPATH 在Path环境…

如何部署Linux AMH服务器管理面板并结合内网穿透远程访问

文章目录 1. Linux 安装AMH 面板2. 本地访问AMH 面板3. Linux安装Cpolar4. 配置AMH面板公网地址5. 远程访问AMH面板6. 固定AMH面板公网地址 AMH 是一款基于 Linux 系统的服务器管理面板&#xff0c;它提供了一系列的功能&#xff0c;包括网站管理、FTP 管理、数据库管理、DNS 管…

如何使用第三方API采集电商数据呢?

电商商家最常唠叨的就是店铺运营难做。每日多平台店铺数据统计汇总繁琐耗时&#xff0c;人工效率偏低&#xff0c;且工作内容有限。 特别是眼下“618&#xff0c;双十一&#xff0c;双十二&#xff0c;年底大促”将至&#xff0c;如何提高运营的效率和质量、保证产品及服务的良…

vue3 使用defineAsyncComponent 动态加载组件

问题场景 在项目中使用静态加载组件基本能覆盖80%的场景了&#xff0c;如下图 但是我们在需要 循环生成一些的component 的时候或者在 开发ssr服务端渲染的页面 就会遇到有些组件以静态方式导入就会报错&#xff0c;导致进程失败&#xff0c;那么这时候就需要用到动态组件。那…

P1808 单词分类

P1808 单词分类 题目描述 Oliver 为了学好英语决定苦背单词&#xff0c;但很快他发现要直接记住杂乱无章的单词非常困难&#xff0c;他决定对单词进行分类。 两个单词可以分为一类当且仅当组成这两个单词的各个字母的数量均相等。 例如 AABAC&#xff0c;它和 CBAAA 就可以…

Vue中对虚拟DOM的理解

作为现代前端开发中的主流框架之一&#xff0c;Vue.js是一个非常流行的JavaScript框架&#xff0c;其核心概念之一就是虚拟DOM&#xff08;Virtual DOM&#xff09;。在本篇文章中&#xff0c;我们将深入探讨Vue中虚拟DOM的概念&#xff0c;并讨论为什么它在前端开发中如此重要…

高清符合要求的SCI图片使用RStudio导出

4.图片格式区别和常识 在计算机中&#xff0c;JPEG&#xff08;发音为jay-peg, IPA&#xff1a;[ˈdʒeɪpɛg]&#xff09;是一种针对照片视频而广泛使用的有损压缩标准方法。这个名称代表Joint Photographic Experts Group&#xff08;联合图像专家小组&#xff09;。此团队创…

总结:图像生成网络

1、最新的几款图像生成网络 eCNN 文献&#xff1a;Bahrami A, Karimian A, Fatemizadeh E, et al. A new deep convolutional neural network design with efficient learning capability: Application to CT image synthesis from MRI[J]. Medical physics, 2020, 47(10): 515…

Linux 分析指定JAVA服务进程所占内存CPU详情

1、获取服务进程PID [rootVM-32-26-centos ~]# service be3Service status Application is running as root (UID 0). This is considered insecure. Running [25383]2、获取进程占用详情 [rootVM-32-26-centos ~]# cat /proc/25383/status Name: java Umask: 0022 State: S…

2024-2-6-复习作业

1> 要求&#xff1a; 源代码&#xff1a; #include <stdio.h> #include <stdlib.h> void output(int arr[],int len) {for(int i0;i<len;i){printf("%d ",arr[i]);}puts(""); } void bubble_sort(int arr[],int len) {for(int i1;i<…

python的进程,线程、协程

python进程的实现 #coding:utf-8 from multiprocessing import Process import timedef run(name):print(%s is running % name)time.sleep(3)print(%s finished his run % name)if __name__ __main__:p Process(targetrun, args(XWenXiang,)) # 创建一个进程对象p.start()…

88 docker 环境下面 前端A连到后端B + 前端B连到后端A

前言 呵呵 最近出现了这样的一个问题, 我们有多个前端服务, 分别连接了对应的后端服务, 前端A -> 后端A, 前端B -> 后端B 但是 最近的时候 却会出现一种情况就是, 有些时候 前端A 连接到了 后端B, 前端B 连接到了 后端A 我们 前端服务使用 nginx 提供前端 html, js…

字符集JAVA

举例&#xff1a; 我们之前在读取文件的时候&#xff0c;文件中都是用英文举例&#xff0c;如果文件内有中文&#xff0c;读取会发生什么 举例&#xff1a;进行读取&#xff0c; //创建字节输入流对象 FileInputStream fisnew FileInputStream("..\\ioDemo\\a.txt"…

市场复盘总结 20240206

仅用于记录当天的市场情况&#xff0c;用于统计交易策略的适用情况&#xff0c;以便程序回测 短线核心&#xff1a;不参与任何级别的调整&#xff0c;采用龙空龙模式 一支股票 10%的时候可以操作&#xff0c; 90%的时间适合空仓等待 今日梯队&#xff1a; 二进三&#xff1a…