C语言小游戏:三子棋

news2024/11/16 2:20:07

目录

🌍前言

🚅目录设计

💎游戏逻辑设置

⚔三子棋棋盘设计

⚔三子棋运行逻辑

👀怎么设置人下棋

👀怎么设置电脑下棋

✈如何判断输赢

✍结语


🌍前言

Hello,csdn的各位小伙伴你们好啊!这次小赵给大家带来的C语言小游戏是三子棋,三子棋也就是大家常玩的井字棋,这样的一个编程想必会让很多人都兴奋不已,好了好了那我们赶快开始吧。

🚅目录设计

目录的设计我们已经讲过好几遍了,还是最简单的那个,简单一个打印*加一些选项。然后用我们的switch函数来实现我们的选择,如输入1进入游戏,0跳出游戏,其他报错然后重新输入。

void meun()
{
	printf("*************************************************\n");
	printf("*********************0.exit**********************\n");
	printf("*********************1.play**********************\n");
	printf("*************************************************\n");
	printf("请输入你的选择:");
}
int main()
{
	int a = 0;
	do
	{
		meun();
		scanf("%d", &a);
		switch (a)
		{
		case 1:
			game();
			break;
		case 0:
			printf("游戏结束\n");
			break;
		default:
			printf("输入错误,请重新输入\n");
			break;
		}
	} while (a);

 在这里我们就不多说了,详细的讲解大家可以看看前面的扫雷和猜数字游戏。(大家可以关注小赵的专栏c语言小游戏哦!

🚀游戏逻辑设置

前面的东西聊完了,那么我们就不得不聊聊我们的游戏逻辑设计,我们得先想明白我们下这个三子棋需要什么,棋盘,棋子对不对,这个至少不能少吧,那下面呢,是不是得看怎样才算赢了这了这盘棋,接着还需要考虑我们和谁下棋得问题,当然小赵在这里剧透一波,这里是和电脑哦!怎么样是不是瞬间对这个代码感到好奇了,那我们抓紧开始吧。

⚔三子棋棋盘设计

首先是棋盘设置,在这里小赵找到了一张我们正常的井字棋的图片。

小赵仿照这个图做了一个我们可以通过编程实现的较为简单的棋盘

​​​​​​​

我们可以注意到这样的一个棋盘就更好实现,因为里面的很多元素是重复的而且可以靠我们|-——来解决。当然这里我们还是和扫雷时候一样进行一个封装,将这个代码装入我们的函数中。

#define ROW  3//定义行
#define COL  3//定义列
void InitBoard(char arr[ROW][COL], int row, int col)
{
	for (int i = 0; i < row; i++)
	{
		for (int j = 0; j < col; j++)
		{
			arr[i][j] = ' ';//对数组初始化,每个位置都是空白
		}
	}
}
void PrintBoard(char arr[ROW][COL], int row, int col)
{
	for (int i = 0; i < row; i++)
	{
		for (int j = 0; j < col; j++)
		{
			printf(" %c ", arr[i][j]);//打印占棋子位子
			if (j < col - 1)
			   printf("|");//打印棋盘元素//   |     |    (最后一个没有)
		}
		printf("\n");//换行
	for (int j = 0; j < col; j++)
	{
		printf("---");
		if (j < col - 1)
		printf("|");//———|———|———(最后一个没有|)
	}
	printf("\n");
	}
}

那么我们的整个棋盘就算结束了,到这里。 

🚀下棋

下面就是下棋了,在这里我们先约定一下我们的棋子是什么,在这里小赵就定义一下*为我们的棋子,那么我们每次下棋实际上就是将这些空白地方放上我们的棋子,也就是取代我们的%c的空白位置。

👀怎么设置人下棋

那么人怎么下棋呢,你完全可以根据小赵的上周的博客那篇教你学C里面讲的二维数组一样,把这个棋盘的下棋位置当成二维数组的每个位置,那么我们在下的时候只需要根据我们的每个坐标来下棋就好了,在这里小赵也是封装成了一个函数。

void PersonSet(char arr[ROW][COL], int row, int col)
{
	   int ret;
	     do
		{
			int x, y;
			scanf("%d %d", &x, &y);
			if (x-1 < row && y-1 < col)
			{
				if (arr[y-1][x-1] == ' ')
				{
					arr[y-1][x-1] = '*';//对第几列第几行下棋
					ret = 0;
				}
			}
			else
			{
				printf("输入错误,请重新输入\n");//报错重新输入
				ret = 1;
			}
		 } while (ret);
}

为了方便大家知道自己的棋是怎么下的,小赵又画了一张图供大家参考,当然画的不好请见谅。

👀怎么设置电脑下棋

这里的电脑下棋,小赵实在也不知道该如何去做一个去能分析对手下棋的强大AL,所以很抱歉,这里小赵用的还是未知数,当然如果大家有什么好的想法可以在小赵的评论区留下留言,让小赵也能学习学习。好了我们接着看电脑下棋,这里为了有所区别,小赵这里给电脑另一个棋字#,让我们有所区分。这里也是封装成一个函数了。

void ComputerSet(char arr[ROW][COL], int row, int col)
{
	srand((unsigned int)time(NULL));
	while (1) 
	{
		int x = rand() % row + 1;
		int y = rand() % col + 1;

		if (arr[y - 1][x - 1] == ' ')
		{
			arr[y - 1][x - 1] = '#';
			break;

		}
	}
}

🚀如何判断输赢

那么这个时候就是最关键的时候了。我们该如何判断输赢了,我们知道井字棋的结局共有三种,第一种是人赢了,第二种电脑赢,第三种平局(即棋盘下满了)。而且还要考虑到什么情况横着竖着和斜着,这里可以说是相当复杂了。我们一步一步来。

✊平局

平局怎么判定,其实就是棋盘上的棋子都下满了,那我们只需要用循环一个个扫就行了,确保棋盘上没有一个空的地方,且这个时候双方没有输赢(这个放在下面)。

int is_FULL(char arr[ROW][COL], int row, int col)
{
	for (int i = 0; i < row; i++)
	{
		for(int j=0;j<col;j++)
			if (arr[i][j] =' ')
			{
				return 0;//如果有空位,返回零
			}
	}
		return 1;//扫完所有放棋子,都没有空位,返回1
}

这里小赵用一种返还数字的方式来判断究竟有木有空的地方,这样我们在下面只需要将判断的三种结局的方式用一个函数穿起来就好了。 

✊判断输赢

在判断输赢这里,我们要尽可能的巧妙和省代码,比如判断人赢还是电脑赢,其实就是判断某一种棋子是否连三个相等。其次就是我们上面的平局用的1/0我们可以加入其中,如若是平局就进入if函数来出来,同时我们还要注意我们这个是一个判断,那么这个判断的时候就要注意我们判断的时候可能不是输赢平,那我们就要继续下,那我们可以再设置一个返还数字在下面,这样经过所有判断都没结果后,我们就输出一个字母意思是继续。

char is_win(char arr[ROW][COL], int row, int col)
{
	int i = 0;
	for (i = 0; i < row; i++)
	{
		if (arr[i][0] ==arr[i][1] && arr[i][1] == arr[i][2] && arr[i][0] != ' ')
		{
			return arr[i][0];//如果一竖行都是一种棋子,则返回这个棋子
		}
	}
	for (i = 0; i < col; i++)
	{
		if (arr[0][i] == arr[1][i] && arr[1][i] == arr[2][i] && arr[0][i] != ' ')
		{
			return arr[0][i];//如果一横行都是一种棋子,则返回这个棋子
		}
	}

	if (arr[0][0] == arr[1][1] && arr[1][1] ==arr[2][2] && arr[1][1] != ' ')
	{
		return arr[1][1];//如果左斜着都是一种棋子,则返回这个棋子
	}

	if (arr[0][2] == arr[1][1] &&arr[1][1] == arr[2][0] && arr[1][1] != ' ')
	{
		return arr[1][1];//如果右斜着都是一种棋子,则返回这个棋子
	}

	//判断平局
	if (is_FULL(arr, row, col))//返回1为真,0为假。
	{
		return 'Q';//如果是平局返还Q
	}

	return 'C';//若上面都不是,返还c意思是游戏继续

}

🛸游戏运行逻辑

我们现在手上已经有了这么多的函数下面就要编我们的整个运行的逻辑吧。首先我们要确定的是我们的判断应该什么时候进行,我觉得应该下一次就判断一次,因为我们的函数中是有继续的选项的,所以完全可以下一次判断一次,同时下一次打印一次。

void game()
{
	char ret;
	char arr[ROW][COL];
	InitBoard(arr, ROW, COL);//定义数组
	PrintBoard(arr, ROW, COL);//打印棋盘
	while (1)
	{
		PersonSet(arr, ROW, COL);//人下
		PrintBoard(arr, ROW, COL);//打印棋盘
		 ret = is_win(arr, ROW, COL);//判断输赢
		if (ret != 'C')//如果输出的不是c,就是不是继续,跳出循环
		{
			
			break;
		}
		printf("电脑下\n");
		ComputerSet(arr, ROW, COL);//电脑下
		PrintBoard(arr, ROW, COL);//打印棋盘
		ret = is_win(arr, ROW, COL);//判断
		if (ret != 'C')
		{
			
			break;
		}
	}
		if (ret == '*')//输出人的棋
		{
			printf("玩家赢\n");
		}
		else if (ret == '#')//输出电脑的棋
		{
			printf("电脑赢\n");
		}
		else//都不是,即使Q
		{
			printf("平局\n");
		}
	
	}

🌍整个代码

最后我们将我们的所有代码像之前的扫雷一样,装在不同的文件中。

game.h

#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<time.h>
#pragma once
#define ROW  3
#define COL  3
void InitBoard(char arr[ROW][COL], int row, int col);
void PrintBoard(char arr[ROW][COL], int row, int col);
void PersonSet(char arr[ROW][COL], int row, int col);
void ComputerSet(char arr[ROW][COL], int row, int col);
char is_win(char arr[ROW][COL], int row, int col);

game.c

#define _CRT_SECURE_NO_WARNINGS 1
#include"game.h"
void meun()
{
	printf("*************************************************\n");
	printf("*********************1.exit**********************\n");
	printf("*********************2.play**********************\n");
	printf("*************************************************\n");
	printf("请输入你的选择:");
}
void game()
{
	char ret;
	char arr[ROW][COL];
	InitBoard(arr, ROW, COL);
	PrintBoard(arr, ROW, COL);
	while (1)
	{
		PersonSet(arr, ROW, COL);
		PrintBoard(arr, ROW, COL);
		 ret = is_win(arr, ROW, COL);
		if (ret != 'C')
		{
			
			break;
		}
		printf("电脑下\n");
		ComputerSet(arr, ROW, COL);
		PrintBoard(arr, ROW, COL);
		ret = is_win(arr, ROW, COL);
		if (ret != 'C')
		{
			
			break;
		}
	}
		if (ret == '*')
		{
			printf("玩家赢\n");
		}
		else if (ret == '#')
		{
			printf("电脑赢\n");
		}
		else
		{
			printf("平局\n");
		}
	
	}

int main()
{
	int a = 0;
	do
	{
		meun();
		scanf("%d", &a);
		switch (a)
		{
		case 2:
			game();
			break;
		case 1:
			printf("游戏结束\n");
			break;
		default:
			printf("输入错误,请重新输入\n");
			break;
		}
	} while (a);
}

✍结语

好了今天的分享就到这里了,大家等会可以试着去敲属于自己的三子棋C语言游戏哦!如果大家喜欢小赵的C语言游戏系列,也可以订阅小赵的C语言小游戏系列,如果大家对这个游戏代码有更好的改进完善方案也可以在评论区留言哦,小赵会一一回复的。

如果觉得文章对你有帮助的话,还请点赞,关注,收藏支持小赵,如有不足还请指点,小赵及时改正,感谢大家支持!!!

​​​​​​​​​​​​​​

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

密码学学习笔记(二十三):哈希函数的安全性质:抗碰撞性,抗第一原象性和抗第二原象性

在密码学中&#xff0c;哈希函数是一种将任意长度的数据映射到固定长度输出的函数&#xff0c;这个输出通常称为哈希值。理想的哈希函数需要具备几个重要的安全性质&#xff0c;以确保数据的完整性和验证数据的来源。这些性质包括抗碰撞性、抗第一原象性和抗第二原象性。 抗碰…

C#网络编程TCP程序设计(Socket类、TcpClient类和 TcpListener类)

目录 一、Socket类 1.Socket类的常用属性及说明 2.Socket类的常用方法及说明 二、TcpClient类 三、TcpListener类 四、示例 1.源码 2.生成效果 TCP(Transmission Control Protocol)是一种面向连接的、可靠的、基于字节流的传输层通信协议。在C#中&#xff0c;TCP程序设…

Fiddler的配置、原理和使用

一、Fiddler的工作原理 本地应用与服务器之间所有的请求&#xff08;request&#xff09;和响应&#xff08;response&#xff09;&#xff0c;由fiddler进行转发&#xff0c;此时fiddler以代理服务器的方式存在。 由于所有的网络数据都要经过fiddler&#xff0c;因此&#xf…

6、原型模式(Prototype Pattern,不常用)

原型模式指通过调用原型实例的Clone方法或其他手段来创建对象。 原型模式属于创建型设计模式&#xff0c;它以当前对象为原型&#xff08;蓝本&#xff09;来创建另一个新的对象&#xff0c;而无须知道创建的细节。原型模式在Java中通常使用Clone技术实现&#xff0c;在JavaSc…

01_W5500简介

目录 W5500简介&#xff1a; 芯片特点: 全硬件TCPIP协议栈: 引脚分布&#xff1a; W5500简介&#xff1a; W5500是一款高性价比的以太网芯片&#xff0c;其全球独一无二的全硬件TCPIP协议栈专利技术&#xff0c;解决了嵌入式以太网的接入问题&#xff0c;简单易用&#xff…

HostHunter虚拟主机发现

HostHunter虚拟主机发现 1.HostHunter2.安装3.参数解释4.实例1.HostHunter HostHunter 一种工具,用于有效发现和提取提供大量目标 IPv4 或 IPv6 地址的主机名。HostHunter 利用简单的 OSINT 和主动协调技术将 IP 目标与虚拟主机名进行映射。这对于发现组织的真正攻击面特别有…

直观清晰的带你了解KMP算法(超详细)

KMP算法用来找某个字符串是否存在某个连续的真子串的 下面举一个例子让抽象的KMP算法更加直观&#xff0c;有助于理解 首先我们要了解KMP算法首先要找到一个next数组来表示主串中每一个字符的回退的下标&#xff08;这个下标是对于真子串而言的&#xff0c;主串不需要回退&…

ubuntu安装tomcat并配置前端项目

1.1查找 # 先更新 sudo apt update # 查找 apt search jdk1.2安装 sudo apt install openjdk-8-jdk1.3验证 java -version 2.安装tomcat 下载链接&#xff1a;Apache Tomcat - Apache Tomcat 8 Software Downloadshttps://tomcat.apache.org/download-80.cgi下载这个&…

layui+ssm实现数据批量删除

layuissm实现数据的批量删除 //数据表格table.render({id: adminList,elem: #adminList,url: ctx "/admin/getAdminList", //数据接口cellMinWidth: 80,even: true,toolbar: #toolbarDemo,//头部工具栏limit: 10,//每页条数limits: [10, 20, 30, 40],defaultToolba…

JS生成登录验证码

采用js生成登录的验证码 采用的技术点有html&#xff0c;css&#xff0c;JS&#xff0c;jQuery HTML&#xff1a; <div class"box_b"><img src"./img/0775639c-c82c-4a29-937f-d2a3bae5151a.png" alt""><div class"regist…

4G基站BBU、RRU、核心网设备

目录 前言 基站 核心网 信号传输 前言 移动运营商在建设4G基站的时候&#xff0c;除了建设一座铁塔之外&#xff0c;更重要的是建设搭载铁塔之上的移动通信设备&#xff0c;这篇博客主要介绍BBU&#xff0c;RRU以及机房的核心网等设备。 基站 一个基站有BBU&#xff0c;…

Leetcode—1038.从二叉搜索树到更大和树【中等】

2023每日刷题&#xff08;四十九&#xff09; Leetcode—1038.从二叉搜索树到更大和树 算法思想 二叉搜索树的中序遍历&#xff08;左根右&#xff09;结果是一个单调递增的有序序列&#xff0c;我们反序进行中序遍历&#xff08;右根左&#xff09;&#xff0c;即可以得到一…

Redis系列之incr和decr命令是线程安全的?

Redis是一个单线程的服务&#xff0c;所以正常来说redis的命令是会排队执行的。incr/decr命令是redis提供的可以实现递增递减的命令&#xff0c;所以这两个命令也是具有原子性的&#xff1f;是线程安全的&#xff1f;这个也是互联网公司面试的常见题&#xff0c;话不多说&#…

linux 命令 tmux 用法详解

一、tmux 解决的痛点&#xff08;screen命令一样可以解决&#xff0c;但是tmux功能更强大&#xff09; 痛点一&#xff1a;大数据传输的漫长一夜 相信做过 Linux 服务运维的同学&#xff0c;都用 scp 进行过服务器间的大文件网络传输。一般这需要很长的时间&#xff0c;这期间…

react结合vant的Dialog实现签到弹框操作

1.需求 有时候在开发的时候&#xff0c;需要实现一个签到获取积分的功能&#xff0c;使用react怎么实现呢&#xff1f; 需求如下&#xff1a; 1.当点击“签到”按钮时&#xff0c;弹出签到框 2.展示签到信息&#xff1a; 签到天数&#xff0c; 对应天数签到能够获取的积分&…

封装时间轴组件 timeline

要求时间轴的点展示进度百分比&#xff0c;线也根据进度不同展示不同长度的颜色 实现效果&#xff1a; 使用的组件库是vant的circle 子组件&#xff1a; <template><div class"m-timeline-area" :style"width: ${width}px"><div class&qu…

XXL-Job详解(五):动态添加、启动任务

目录 前言XXL-Job API接口添加任务API动态添加任务动态启动任务 前言 看该文章之前&#xff0c;最好看一下之前的文章&#xff0c;比较方便我们理解 XXL-Job详解&#xff08;一&#xff09;&#xff1a;组件架构 XXL-Job详解&#xff08;二&#xff09;&#xff1a;安装部署 X…

SQLserver通过字符串中间截取然后分组

当我们存的数据是json的时候可以全部取出在模糊查询但是有多个重复数据的时候就没办法准确的模糊出来这个时候我们就需要用的字符串截取 --创建函数create FUNCTION [dbo].[Fmax] (str varchar(50),start VARCHAR(50),length VARCHAR(50)) RETURNS varchar(max) AS BEGINDEC…

如何使用Cloudreve搭建本地云盘系统并实现随时远程访问

文章目录 1、前言2、本地网站搭建2.1 环境使用2.2 支持组件选择2.3 网页安装2.4 测试和使用2.5 问题解决 3、本地网页发布3.1 cpolar云端设置3.2 cpolar本地设置 4、公网访问测试5、结语 1、前言 自云存储概念兴起已经有段时间了&#xff0c;各互联网大厂也纷纷加入战局&#…

nginx部署和安装-后端程序多端口访问-后端代理设置

部分补充 查看nginx是否安装http_ssl_module模块 ./nginx -V 看到有 configure arguments: --with-http_ssl_module, 则已安装。 如果没有安装&#xff1a;参考文档 nginx官网地址&#xff1a;nginx: download 这里下载nginx-1.18.0稳定版tar.gz 下载后&#xff0c;利用…