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

文章目录

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

前言

三子棋是一种民间传统游戏,又叫九宫棋、圈圈叉叉棋、一条龙、井字棋等。游戏分为双方对战,双方依次在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, 不满返回0if (Is_Full(board, row, col)){return 'Q';}/* 继续 */return 'C';
}

四、结束

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

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

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

构建安全可靠的系统:第十六章到第二十章

第四部分&#xff1a;维护系统 原文&#xff1a;Part IV. Maintaining Systems 译者&#xff1a;飞龙 协议&#xff1a;CC BY-NC-SA 4.0 准备应对不舒适情况的组织有更好的机会处理关键事件。 尽管不可能为可能扰乱您组织的每种情况制定计划&#xff0c;但作为综合灾难规划策略…

dubbo的springboot集成

1.什么是dubbo&#xff1f; Apache Dubbo 是一款 RPC 服务开发框架&#xff0c;用于解决微服务架构下的服务治理与通信问题&#xff0c;官方提供了 Java、Golang 等多语言 SDK 实现。使用 Dubbo 开发的微服务原生具备相互之间的远程地址发现与通信能力&#xff0c; 利用 Dubbo …

近两年最火的图像处理算法

近两年&#xff08;2022-2023年间&#xff09;在图像处理领域&#xff0c;有几个算法和技术特别受到关注。这些技术在提高图像质量、加速处理过程以及提升图像识别的准确性方面取得了显著进展。 以下是一些显著的例子&#xff1a; 生成对抗网络&#xff08;GANs&#xff09; GA…

基于sprinmgboot实习管理系统源码和论文

随着信息化时代的到来&#xff0c;管理系统都趋向于智能化、系统化&#xff0c;实习管理也不例外&#xff0c;但目前国内仍都使用人工管理&#xff0c;市场规模越来越大&#xff0c;同时信息量也越来越庞大&#xff0c;人工管理显然已无法应对时代的变化&#xff0c;而实习管理…

基于SELinux三权分立配置方法

1.系统安装 系统安装完成后,系统当前的SELinux配置为: # cat /etc/selinux/config SELINUX=enforcing SELINUXTYPE=targeted 2.SELinux环境准备 # yum install setools policycoreutils.x86_64 selinux-policy-mls.noarch setroubleshoot.x86_64 setools-console -y 3.SELin…

共享经济风潮下的国际化之路:品牌出海的机遇与挑战解析

近年来&#xff0c;共享经济在全球范围内迅速崛起&#xff0c;成为商业模式的新风口。随着这一趋势的发展&#xff0c;许多品牌开始积极出海&#xff0c;将共享理念带到国际市场。共享经济的全球化为品牌提供了前所未有的机遇&#xff0c;然而&#xff0c;也伴随着一系列的挑战…

【昕宝爸爸系列】如何将集合变成线程安全的?

如何将集合变成线程安全的? ✅典型解析&#x1f7e2;拓展知识仓☑️Java中都有哪些线程安全的集合&#xff1f;&#x1f7e0;线程安全集合类的优缺点是什么&#x1f7e1;如何选择合适的线程安全集合类☑️如何解决线程安全集合类并发冲突问题✔️乐观锁实现方式 (具体步骤)。✅…

Vue-9、Vue事件修饰符

1、prevent 阻止默认事件 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><title>事件修饰符</title><!--引入vue--><script type"text/javascript" src"https://cdn.jsdeliv…

烟火检测AI边缘计算智能分析网关V4在安防项目中的应用及特点

一、行业背景 随着社会和经济的发展&#xff0c;公共安全和私人安全的需求都在不断增长。人们需要更高效、更准确的安防手段来保障生命财产安全&#xff0c;而人工智能技术正好可以提供这种可能性&#xff0c;通过智能监控、人脸识别、行为分析等手段&#xff0c;大大提高了安防…

【C++进阶05】AVL树的介绍及模拟实现

一、AVL树的概念 二叉搜索树的缺点 二叉搜索树虽可以缩短查找效率 但如果数据有序或接近有序 二叉搜索树将退化为单支树 查找元素相当于在顺序表中搜索元素&#xff0c;效率低下 AVL树便是解决此问题 向二叉搜索树中插入新结点 并保证每个结点的左右子树 高度之差的绝对值不超…

Appium + ios环境搭建过程Mac

前提&#xff1a; 已经搭建好NodeJavaPythonAppium...环境 见下面的文章&#xff1a; ok的话按照下面的步骤搭建IOs的自动化 1. 安装Xcode 官方下载 (Downloads and Resources - Xcode - Apple Developer 1)AppStore 下载安装最新版本 2. 依赖工具 工具名描述libimobile…

springboot 房屋租赁系统

spring boot mysql mybatis 前台后端