C语言之三子棋小游戏的应用

news2024/11/16 13:39:40

文章目录

  • 前言
    • 一、前期准备
      • 模块化设计
    • 二、框架搭建
    • 三、游戏实现
      • 打印棋盘
      • 代码优化
      • 玩家下棋
      • 电脑下棋
      • 判断输赢
    • 四、结束

前言

三子棋是一种民间传统游戏,又叫九宫棋、圈圈叉叉棋、一条龙、井字棋等。游戏分为双方对战,双方依次在9宫格棋盘上摆放棋子,率先将自己的三个棋子走成一条线就视为胜利,而对方就算输了,但是三子棋在很多时候会出现和棋的局面。

本篇博客就来进行讲解这个三子棋小游戏,跟着我来一起看把!(本文使用的编译器是VS2022

一、前期准备

模块化设计

在写三子棋的时候,我们先要了解一下什么事模块化设计:

模块化程序设计是指在进行程序设计时将一个大程序按照功能划分为若干小程序模块,每个小程序模块完成一个确定的功能,并在这些模块之间建立必要的联系,通过模块的互相协作完成整个功能的程序设计方法

  • 上面是百度百科的介绍,可能有同学看不懂,简单来说就是份文件写
  • 在我们写一些程序的时候就会遇到一个.c文件里写很多,会显得杂乱,可读性会变的非常差,那么我们就要使用份文件来写代码,这样就会变得条理清晰,可读性强,这样是一种良好的编程习惯,那么怎么做呢?接下来看~~
  1. 建立一个game.h头文件:存储行列信息,包含函数库,对函数进行声明
  2. 建立一个game.c文件:实现游戏中的函数
  3. 建立一个test.c文件:实现函数主体逻辑,在书写时可用此函数进行测试
  4. game.ctest.c文件中包含#include"game.h"

在这里插入图片描述

二、框架搭建

创建好文件后,将game.c和test.c引入game.h,头文件的包含和函数的声明就在这里面

在这里插入图片描述

游戏界面:

在这里插入图片描述

game.h

  • 这里定义一个三行三列,并且初始化,当想要变成n行m列的只需要改一下这里define定义的就行
//行
#define ROW 3
//列
#define COL 3
//初始化棋盘
void InitBoard(char board[ROW][COL],int row,int col);
  • 玩家输入选择,switch处理对应逻辑,输入值顺便还可以作为循环结束的条件。

test.c

#define _CRT_SECURE_NO_WARNINGS 1
#include"game.h"

void menu()
{
	printf("\n");
	printf("**************************\n");
	printf("*****     1.play    ******\n");
	printf("*****     0.exit    ******\n");
	printf("**************************\n");
	printf("\n");
}

void game()
{
	printf("玩游戏\n");
}

int 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.h

  • 在这里把需要引入的头文件写入
#pragma once
#include<stdio.h>
  • 可以看到,游戏已经正常运行了,但是里面的game函数还没有实现,接下来就让我们继续往下看(完成一部分功能就运行一下看看,及时发现BUG,越早发现越容易找到BUG)

三、游戏实现

创建棋盘&初始化棋盘

game.h

void InitBoard(char board[ROW][COL],int row,int col);

test.c

void game()
{
	//创建棋盘
	char board[ROW][COL];
	//初始化棋盘
	InitBoard(board, ROW, COL);
}

game.c

  • 初始化棋盘,将数组所有元素初始化为空格
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] = ' ';//初始化为空格
		}
	}
}

打印棋盘

game.h
打印棋盘

void DisplayBoard(char board[ROW][COL], int row,int col);

test.c

DisplayBoard(board, ROW, COL);

game.c

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

在这里插入图片描述
如果我们要修改棋盘大小,行是循环出来的,但是列就写死了

代码优化

  • 首先打印空格 空格|,要打印row行col列,这里要注意的是当col列为col-1时才打印,也就是说打印了2列|
  • 打印---也是一样的,同理

game.c

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("|");
		}
		printf("\n");
		if (i < row - 1)
		{
			for (j = 0; j < col; j++)
			{
				printf("---");
				if (j < col - 1)
					printf("|");
			}
			printf("\n");
		}
	}
}

我们将ROW和COL修改成10也是可以打印的

在这里插入图片描述

玩家下棋

game.h

void PlayMove(char board[ROW][COL], int row, int col);

test.c

//玩家下棋
while (1)
{
	PlayMove(board, ROW, COL);
	DisplayBoard(board, ROW, COL);
}
  • 玩家下棋是不是要输入坐标,那么我们就先定义x和y,首先判断玩家输入的xy坐标合法,在棋盘范围内,如果合法,就继续,否则提示
  • 在玩家下棋时,需要判断是否要下的位置为空格,是空格说明当前位置没有棋子,不是空格说明当前位置已被下棋,就提示重新下棋

game.c

void PlayMove(char board[ROW][COL], int row, int col)
{
	int x = 0;
	int 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");
		}
	}
}

电脑下棋

  • 电脑下棋要进行

game.h

#include<time.h>
#include<stdlib.h>
void ComputerMove(char board[ROW][COL], int row, int col);

test.c
在main函数里,调用srand

srand((unsigned int)time(NULL));

game函数

while (1)
{
	//玩家下棋
	PlayMove(board, ROW, COL);
	DisplayBoard(board, ROW, COL);
	//电脑随机下棋
	ComputerMove(board, ROW, COL);
	DisplayBoard(board, ROW, COL);
}

game.c

  • 电脑下棋也是同理,调用rand函数随机生成一个数
  • 检测要下棋的位置是否为空格,是空格才可以下,不是空格重新生成一个随机数,重新下棋
void ComputerMove(char board[ROW][COL], int row, int col)
{
	printf("电脑下棋\n");
	int x = 0;
	int y = 0;
	x = rand() % row;
	y = rand() % col;
	while (1)
	{
		if (board[x][y] == ' ')
		{
			board[x][y] = '#';
			break;
		}
	}
}

这个时候就可以正常下棋了,但是:没有判断输赢,下完了也不会结束,而是死循环

判断输赢

判断输赢有四种状态

  • 玩家赢
  • 电脑赢
  • 平局
  • 游戏继续

玩家赢返回*
电脑赢返回#
平局返回Q
游戏继续返回C

game.h

char IsWin(char board[ROW][COL], int row, int col);
  • 这里判断输赢的时候首先玩家下棋,进行判断有没有输赢,然后电脑下棋,如果有一方输赢了,就进行返回

test.c

char ret = 0;
while (1)
{
	//玩家下棋
	PlayMove(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
{
	pprintf("平局\n");
}
  • 这里是进行判断棋盘输赢的逻辑

game.c

int Is_Full(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 1; // 棋盘满了
}


char IsWin(char board[ROW][COL], int row, int col)
{
	int i = 0;
	/* 判断三行 */
	for (i = 0; i < row; i++)
	{
		if (board[i][0] == board[i][1] && board[i][1] == board[i][2] && board[i][1] != ' ')
		{
			return  board[i][1];
		}
	}

	/* 判断三列 */
	for (i = 0; i < col; i++)
	{
		if (board[0][i] == board[1][i] && board[1][i] == board[2][i] && board[1][i] != ' ')
		{
			return board[1][i];
		}
	}

	/* 判断对角线 */
	if (board[0][0] == board[1][1] && board[1][1] == board[2][2] && board[1][1] != ' ')
	{
		return board[1][1];
	}
	if (board[0][2] == board[1][1] && board[1][1] == board[2][0] && board[1][1] != ' ')
	{
		return board[1][1];
	}

	/* 判断平局 */
	//如果棋盘满了返回1, 不满返回0
	if (Is_Full(board, row, col))
	{
		return 'Q';
	}

	/* 继续 */
	return 'C';
}

四、结束

最后代码还是可以优化的,比如判断输赢这里是写死了,只能判断三行三列斜线,如果是多行就不能了,还有让电脑下棋智能一点,能判断玩家下棋的位置再进行下棋,这样更有可玩性!

好了,本文就到这里结束了,下一篇是扫雷小游戏!

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

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

相关文章

chatglm3的api调用

conda activate chatglm3 cd openai_api_demo python openai_api.py 启动ok&#xff0c;然后内网映射后 anaconda启动jupyter !pip install openai1.6.1 -i https://pypi.tuna.tsinghua.edu.cn/simple/ """ This script is an example of using the OpenAI …

uniapp自定义底部导航栏

1.新建 nav-custom.vue组件 <template><view class"nav-box" :style"{height:heightpx,background:bgColor}"><!-- 自定义导航栏 --><view class"status_bar" :style"{height:statusBarHeightpx}"><!-- u…

rke2 Offline Deploy Rancher v2.8.0 latest (helm 离线部署 rancher v2.8.0)

文章目录 预备条件为什么是三个节点&#xff1f;​预备条件配置私有仓库介质清单安装 helm安装 cert-manager下载介质镜像入库helm 部署卸载 安装 rancher镜像入库helm 安装 验证 预备条件 所有支持的操作系统都使用 64-bit x86 架构。Rancher 兼容当前所有的主流 Linux 发行版…

export default 和exprot

1.默认导入和默认导出 语法: export default {需要输出的内容} 接收: import 成员变量的名字 from 文件夹的路径 案例&#xff1a; a.mjs文件夹下默认导出 export default{a:10,b:20,show(){console.log(123);} } 在b.mjs文件中用成员变量进行接收 import AA from &q…

手撕单链表(单向,不循环,不带头结点)的基本操作

&#x1d649;&#x1d65e;&#x1d658;&#x1d65a;!!&#x1f44f;&#x1f3fb;‧✧̣̥̇‧✦&#x1f44f;&#x1f3fb;‧✧̣̥̇‧✦ &#x1f44f;&#x1f3fb;‧✧̣̥̇:Solitary-walk ⸝⋆ ━━━┓ - 个性标签 - &#xff1a;来于“云”的“羽球人”。…

5,sharding-jdbc入门-sharding-jdbc广播表

执行sql #在数据库 user_db、order_db_1、order_db_2中均要建表 CREATE TABLE t_dict (dict_id BIGINT (20) NOT NULL COMMENT 字典id,type VARCHAR (50) CHARACTER SET utf8 COLLATE utf8_general_ci NOT NULL COMMENT 字典类型,code VARCHAR (50) CHARACTER SET utf8 COLLAT…

windows和liunx对比及Linux分类

windows一定比liunx差吗&#xff0c;这绝对是天大误解&#xff0c;不是说你常用的开始是liunx就代表windows差 windows和liunx对比 有人说Linux性能远高于Windows&#xff0c;这个笔者是不认可的&#xff0c;给Linux套上一个图形界面&#xff0c;你再使劲美化一下&#xff0c…

【论文精读】A Survey on Large Language Model based Autonomous Agents

A Survey on Large Language Model based Autonomous Agents 前言Abstract1 Introduction2 LLM-based Autonomous Agent Construction2.1 Agent Architecture Design2.1.1 Profiling Module2.1.2 Memory ModuleMemory StructuresMemory FormatsMemory Operations 2.1.3 Plannin…

Android Canvas图层saveLayer剪切clipPath原图addCircle绘制对应圆形区域并放大,Kotlin(3)

Android Canvas图层saveLayer剪切clipPath原图addCircle绘制对应圆形区域并放大&#xff0c;Kotlin&#xff08;3&#xff09; 在文章2 Android Canvas图层saveLayer剪切clipPath原图addCircle绘制对应圆形区域&#xff0c;Kotlin&#xff08;2&#xff09;-CSDN博客 的基础上&…

上海雏鸟科技无人机灯光秀跨年表演点亮三国五地夜空

2023年12月31日晚&#xff0c;五场别开生面的无人机灯光秀跨年表演在新加坡圣淘沙、印尼雅加达、中国江苏无锡、浙江衢州、陕西西安等五地同步举行。据悉&#xff0c;这5场表演背后均出自上海的一家无人机企业之手——上海雏鸟科技。 在新加坡圣淘沙西乐索海滩&#xff0c;500架…

【Python学习】Python学习11-元组

目录 【Python学习】Python学习11-元组 前言创建语法创建语法特殊形式访问元组操作元组元组运算符元组内置函数Python列表函数&方法参考 文章所属专区 Python学习 前言 本章节主要说明Python的Python 的元组与列表类似&#xff0c;不同之处在于元组的元素不能修改。通过小…

我的 AI 成长星球,邀请你加入

大家好啊&#xff0c;我是董董灿。 2023年终总结时我这个小白坚持写作一整年&#xff0c;赚了多少&#xff1f;提到了一点&#xff0c;2024希望自己创建一个免费星球。 其实一直就想弄一个高质量的 AI 知识交流平台&#xff0c;方便大家一起交流和学习&#xff0c;同时提高对 …

Python虚拟环境轻松配置:Jupyter Notebook中的内核管理指南

问题 在Python开发中&#xff0c;一些人在服务器上使用Jupyter Notebook中进行开发。一般是创建虚拟环境后&#xff0c;向Jupyter notebook中添加虚拟环境中的Kernel&#xff0c;后续新建Notebook中在该Kernel中进行开发&#xff0c;这里记录一下如何创建Python虚拟环境以及添…

关于Vue前端接口对接的思考

关于Vue前端接口对接的思考 目录概述需求&#xff1a; 设计思路实现思路分析1.vue 组件分类和获取数值的方式2.http 通信方式 分类 如何对接3.vue 组件分类和赋值方式&#xff0c; 参考资料和推荐阅读 Survive by day and develop by night. talk for import biz , show your p…

C/C++调用matlab

C/C调用matlab matlab虽然可以生成C/C的程序&#xff0c;但其能力很有限&#xff0c;很多操作无法生成C/C程序&#xff0c;比如函数求解、优化、拟合等。为了解决这个问题&#xff0c;可以采用matlab和C/C联合编程的方式进行。使用matlab将关键操作打包成dll环境&#xff0c;再…

仿蓝奏云网盘 /file/list SQL注入漏洞复现

0x01 产品简介 仿蓝奏网盘是一种类似于百度网盘的文件存储和共享解决方案。它为用户提供了一个便捷的平台,可以上传、存储和分享各种类型的文件,方便用户在不同设备之间进行文件传输和访问。 0x02 漏洞概述 仿蓝奏云网盘 /file/list接口处存在SQL注入漏洞,登录后台的攻击…

代码随想录day20 开始二叉搜索树

654.最大二叉树 题目 给定一个不含重复元素的整数数组。一个以此数组构建的最大二叉树定义如下&#xff1a; 二叉树的根是数组中的最大元素。左子树是通过数组中最大值左边部分构造出的最大二叉树。右子树是通过数组中最大值右边部分构造出的最大二叉树。 通过给定的数组构…

零基础学习数学建模——(二)数学建模的步骤

本篇博客将详细介绍数学建模的步骤。 文章目录 引例&#xff1a;年夜饭的准备第一步&#xff1a;模型准备第二步&#xff1a;模型假设第三步&#xff1a;模型建立第四步&#xff1a;模型求解第五步&#xff1a;结果分析第六步&#xff1a;模型检验第七步&#xff1a;模型应用及…

Kubernets(K8S)启动和运行01 快速入门

简介 Kubernetes is an open source orchestrator for deploying containerized applications. It was originally developed by Google, inspired by a decade of experience deploying scalable, reliable systems in containers via application-oriented APIs. Kubernete…

C语言基础语法跟练 day2

题源&#xff1a;牛客网 16、BoBo写了一个十六进制整数ABCDEF&#xff0c;他问KiKi对应的十进制整数是多少。 #include <stdio.h>int main() { //创建变量char arr[] "ABCDEF";int i;int sum0,c; //依次转换十六进制为十进制for(i0; arr[i]!\0; i){char b …