大二必做项目贪吃蛇超详解之中篇游戏设计与分析

2024-09-01 18:28

本文主要是介绍大二必做项目贪吃蛇超详解之中篇游戏设计与分析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

贪吃蛇系列文章

上篇win32
中篇设计与分析

文章目录

  • 贪吃蛇系列文章
  • 1. 地图
    • 1. 1 `<locale.h>`本地化
    • 1. 2 类项
    • 1. 3 setlocale函数
    • 1. 4 宽字符的打印
    • 1. 5 地图坐标
  • 2. 蛇身和食物
  • 3. 数据结构设计
  • 4. 游戏流程设计
  • 5. 核心逻辑实现分析
    • 5. 1 游戏主逻辑
    • 5. 2 GameStart
      • 5. 2. 1 SetInit
      • 5. 2. 2 Welcome
      • 5. 2. 3 InitMap
      • 5. 2. 4 InfoPrint
      • 5. 2. 5 SnakeInit
      • 5. 2. 6 CreatFood


1. 地图

我们最终的贪吃蛇界面是这个样子,可以发现这和之前写的C语言项目的最大不同就在于文字不是依次排列的,那我们的地图应该如何布置呢?
1
2
3
这里回顾一下控制台窗口的一些知识,如果想在控制台的窗口中指定位置输出信息,我们得知道该位置的坐标,所以首先介绍一下控制台窗口的坐标知识。
控制台窗口的坐标如下所示,横向的是X轴,从左向右依次增长,纵向是Y轴,从上到下依次增长
1
在游戏地图上,我们打印墙体使用宽字符□,打印蛇使用宽字符●,打印食物使用宽字符★(这些字符都可以在输入法中打出来)
普通的字符是占一个字节的,这类宽字符是占用2个字节
这里再简单的讲一下C语言的国际化特性相关的知识,过去C语言并不适合非英语国家(地区)使用,因为C语言最初假定字符都是单字节的。但是这些假定并不是在世界的任何地方都适用。

C语言字符默认是采用ASCI编码的,ASCI字符集采用的是单字节编码,且只使用了单字节中的低7位,最高位是没有使用的,可表示为0xxxxxxx;可以看到,ASCII字符集共包含128个字符,在英语国家中,128个字符是基本够用的,但是,在其他国家语言中,比如,在法语中,字母上方有注音符号,它就无法用 ASCI码表示。于是,一些欧洲国家就决定,利用字节中闲置的最高位编入新的符号。比如,法语中的é的编码为130(二进制10000010)。这样一来,这些欧洲国家使用的编码体系,可以表示最多256个符号。但是,这里又出现了新的问题。不同的国家有不同的字母,因此,哪怕它们都使用256个符号的编码方式,代表的字母却不一样。比如,130在法语编码中代表了é,在希伯来语编码中却代表了字母Gimel(汉),在俄语编码中又会代表另一个符号。但是不管怎样,所有这些编码方式中,0–127表示的符号是一样的,不一样的只是128–255的这一段。至于亚洲国家的文字,使用的符号就更多了,汉字就多达10万左右。一个字节只能表示256种符号,肯定是不够的,就必须使用多个字节表达一个符号。比如,简体中文常见的编码方式是 GB2312,使用两个字节表示一个汉字,所以理论上最多可以表示 256x256=65536 个符号。

后来为了使C语言适应国际化,C语言的标准中不断加入了国际化的支持。比如:加入了宽字符的类型wchar_t宽字符的输入和输出函数,加入了<locale.h>头文件,其中提供了允许程序员针对特定地区(通常是国家或者说某种特定语言的地理区域)调整程序行为的函数。

1. 1 <locale.h>本地化

<locale.h>提供的函数用于控制C标准库中对于不同的地区会产生不一样行为的部分。
在标准中,依赖地区的部分有以下几项:

数字量的格式
货币量的格式
字符集
日期和时间的表示形式

1. 2 类项

通过修改地区,程序可以改变它的行为来适应世界的不同区域。但地区的改变可能会影响库的许多部分,其中一部分可能是我们不希望修改的。所以C语言支持针对不同的类项进行修改,下面的一个宏指定一个类项:
LC_COLLATE:影响字符串比较函数 strcoll()strxfrm()
LC_CTYPE:影响字符处理函数的行为。
LC_MONETARY:影响货币格式
LC_NUMERIC:影响 printf()的数字格式。
LC_TIME:影响时间格式 strftime()wcsftime()
LC_ALL:针对所有类项修改,将以上所有类别设置为给定的语言环境。

微软开发文档对类项的介绍

1. 3 setlocale函数

char*setlocale(int category,const char* locale);

setlocale 函数用于修改当前地区,可以针对一个类项修改,也可以针对所有类项。
setlocale 的第一个参数可以是前面说明的类项中的一个,那么每次只会影响一个类项,
如果第一个参数是LC_ALL,就会影响所有的类项。
C标准给第二个参数仅定义了2种可能取值:"C"(正常模式)和""(空字符串,本地模式)。
在任意程序执行开始,都会隐藏式执行调用:

setlocale(LC ALL,"C");

当地区设置为"C"时,设置为C语言默认的模式,这时库函数按正常方式执行。
当程序运行起来后如果想改变地区,就需要调用setlocale函数。用""作为第2个参数,调用setlocale函数就可以切换到本地模式,这种模式下程序会适应本地环境。
比如:切换到我们的本地模式(汉字是宽字符)后就支持宽字符的输出等

setlocale(LC_ALL,"");//切换到本地环境

setlocale 的返回值是一个字符串指针,表示已经设置好的格式。如果调用失败,则返回空指针NULL
setlocale也可以用来查询当前地区,第二个参数设为NULL就可以了。

#include <locale.h>
#include<stdio.h>
int main()
{char* loc;loc = setlocale(LC_ALL, NULL);printf("默认的本地信息:%s\n", loc);loc = setlocale(LC_ALL, "");printf("设置后的本地信息: %s\n", loc);return 0;
}

1

1. 4 宽字符的打印

那如果想在屏幕上打印宽字符,怎么打印呢?
宽字符的字面量必须加上前缀L,否则C语言会把字面量当作窄字符类型处理。
前缀L在单引号前面,表示宽字符,宽字符的打印使用 wprintf,对应 wprintf()的占位符为 %lc
前缀L在双引号前面,表示宽字符串,对应 wprintf()的占位符为 %ls

#include <stdio.h>
#include<locale.h>
int main() {setlocale(LC_ALL, "");wchar_t ch1 = L'●';wchar_t ch2 = L'微';	//汉字也是宽字符wchar_t ch3 = L'软';	wchar_t ch4 = L'★';printf("ab\n");wprintf(L"%lc\n", ch1);	//不要忘记带Lwprintf(L"%lc\n", ch2);wprintf(L"%lc\n", ch3);wprintf(L"%lc\n", ch4);return 0;
}

2
从输出的结果来看,我们发现一个普通字符占一个字符的位置,但是打印一个汉字字符或者宽字符,占用2个字符的位置,那么我们如果要在贪吃蛇中使用宽字符,就得计算好坐标,让X坐标一直为偶数,不然会出现一些问题。
3

1. 5 地图坐标

我们以实现一个棋盘27行,58列的棋盘分析,再围绕地图画出墙,
如下:
4

2. 蛇身和食物

初始化状态:假设蛇的长度是5,蛇身的每个节点是●,在固定的一个坐标处,比如(24,5)处开始出现蛇,连续5个节点。
注意:蛇的每个节点的x坐标必须是2个倍数,否则可能会出现蛇的一个节点有一半儿出现在墙体中,另外一般在墙外的现象,坐标不好对齐。
关于食物,就是在墙体内随机生成一个坐标(x坐标必须是2的倍数),坐标不能和蛇的身体重合,然后打印★。
2

3. 数据结构设计

在游戏运行的过程中,蛇每次吃一个食物,蛇的身体就会变长一节,那么使用链表存储蛇的信息就比较方便了,蛇的每一节其实就是链表的每个节点。每个节点只要记录好蛇身节点在地图上的坐标就行所以蛇节点结构如下:

typedef struct SnakeNode
{int x;int y;struct SnakeNode* next;
}SnakeNode,* pSnakeNode;

要管理整条贪吃蛇,我们再封装一个Snake的结构来维护整条贪吃蛇:
那么Snake中应该有哪些数据呢?

  1. 作为一个链表,头结点是必须要保存下来的
  2. 贪吃蛇可以改变方向,那么贪吃蛇的方向也应该要存储下来
  3. 如果我们想判断贪吃蛇是否死亡,并在游戏结束时告诉玩家是如何死亡的,可以把游戏状态也存储起来
  4. 在游戏中当然少不了分数
  5. 每次吃食物的分数,这个会随着速度的改变而改变,所以也要存储起来
  6. 食物的位置,这个贪吃蛇每次只会在场上生成一个食物,将食物的信息放在Snake中,可以方便开发
  7. 睡眠时间,这个实际上是速度,我们在游戏运行函数中再介绍

另外可以发现,方向只有四个,可以一一列举出来,所以我们可以使用枚举

enum DERCTION	//方向
{UP = 1, DOWN,LEFT,RIGHT
};

状态实际上也是有限的:正常,撞墙,撞到自己,玩家自行退出,也可以一一列举:

enum STATUS
{NORMAL,KILL_BY_WALL,KILL_BY_SELF,ESC
};

那么Snake结构体就可以写成这样,在变量名称前加上_方便与外部变量区分

typedef struct Snake
{pSnakeNode _Head;	//头enum DERCTION _Dir;	//方向enum STATUS _Sta;	//状态pSnakeNode _Food;	//食物int _FoodAdd;		//食物加的分数int _Score;			//当前分数int _SleepTime;		//睡眠时间
}Snake,*pSnake;

4. 游戏流程设计

5
那么至此,前期准备基本完成,接下来我们开始完成游戏的核心逻辑

5. 核心逻辑实现分析

5. 1 游戏主逻辑

程序开始就设置程序支持本地模式,然后进入游戏的主逻辑。
主逻辑分为3个过程

游戏开始(GameStart)完成游戏的初始化
游戏运行(GameRun)完成游戏运行逻辑的实现
游戏结束(GameOver)完成游戏结束的工作

注意:setlocale(LC_ALL, "");不需要放在上面的逻辑中,因为上面的逻辑会随着游戏的再来一把反复执行,而这个代码并不需要反复运行。

<test.h>

#include"game.h"void game()
{char input = 'y';		//用于判断是否再来一把do{Snake s = { 0 };	//做出一条蛇,将其中的内容都置为空srand((unsigned int)time(NULL));	//食物的生成需要随机数,我们在这里设置一下//开始游戏GameStart(&s); //进行游戏GameRun(&s);//结束游戏GameOver(&s); //这个代码用于解决一个bug,在后面介绍//这是AI给出的解决办法,就不多介绍了,<conio.h>是这两个函数需要的头文件//这个while循环是用来读取蛇运行的时候按下的VK虚拟键的循环,//把在蛇运行的时候按下的VK键的键值全面读走(包括上键的键值)while (_kbhit())	//_kbhit()检测是否有按键被按下{//使用 _getch() 获取按下的键_getch();}//如果是主动退出的,就不需要询问是否再来一把了if (s._Sta == ESC){input = 'n';getchar();	//这个getchar用于在release版本下阻止程序直接退出}//在结束之后,询问是否要再来一把else{SetPos(15, 15);printf("要再来一把吗?(Y/y)");input = getchar();while (getchar() != '\n');	//清理'\n'}} while (input=='y'||input=='Y');SetPos(10, 27);		//程序退出时,会有一个xxx程序已正常退出的提示,我们让它不要破坏游戏地图
}int main()
{setlocale(LC_ALL, "");//设置能输出长字符game();
}

在游戏过程中我们会用到非常多次SetPos来设置光标位置,至于这些位置的具体坐标可以自行不断尝试来找到较好的位置,博客中的是我个人觉得比较好的。

5. 2 GameStart

这个部分要完成的任务:

控制台窗口大小的设置
控制台窗口名字的设置
鼠标光标的隐藏
打印欢迎界面
创建地图
初始化蛇
创建第一个食物

我们将其中的每一个任务分别封装成一个函数:

void GameStart(pSnake ps)
{//设置控制台大小,隐藏光标SetInit();//打印欢迎界面Welcome();//布置地图InitMap();//打印介绍信息InfoPrint();//放蛇SnakeInit(ps);//放食物CreatFood(ps);//getchar();	//可以用来停止代码执行,方便调试,项目完成后要注释掉
}

5. 2. 1 SetInit

void SetInit()
{system("mode con cols=100 lines=30");system("title 贪吃蛇");HANDLE Houtput = GetStdHandle(STD_OUTPUT_HANDLE);CONSOLE_CURSOR_INFO stdoutinfo;GetConsoleCursorInfo(Houtput, &stdoutinfo);stdoutinfo.bVisible = false;SetConsoleCursorInfo(Houtput, &stdoutinfo);
}

这个函数就是上一篇文章的主要内容,这里就不再赘述了。

5. 2. 2 Welcome

void Welcome()
{//打印欢迎界面SetPos(40, 15);wprintf(L"欢迎来到贪吃蛇");SetPos(40, 25);system("pause");	//这个代码相当于打印一个"请按任意键继续...",和 getchar();system("cls");		//清空屏幕SetPos(25, 12);wprintf(L"按↑↓←→控制方向,F1加速,F2减速,速度越快,分数越高");	//打印汉字也可以使用 printfSetPos(25, 13);wprintf(L"空格键暂停,ESC退出");SetPos(40, 25);system("pause");system("cls");
}

这个函数就是游戏最开始的两个界面。

5. 2. 3 InitMap

我们在这个函数中会用许多次宽字符,为了方便使用,我们可以在头文件中进行宏定义

#define WALL L'□'
#define SNAKE_BODY L'●'
#define FOOD L'★'

这样,比如我们要打印墙体,我们就可以直接:

wprintf(L"%lc",WALL);

参考代码:

void InitMap()
{for (int i = 0; i < 60; i += 2)wprintf(L"%lc",WALL);	//打印第一行SetPos(0, 28 - 1);for (int i = 0; i < 60; i += 2)wprintf(L"%lc", WALL);	//打印最下面一行for (int i = 1; i < 28; i++){SetPos(0, i);wprintf(L"%lc", WALL);	//打印左边一列}for (int i = 1; i < 28; i++){SetPos(60 - 2, i);wprintf(L"%lc", WALL);	//打印右边一列}
}

5. 2. 4 InfoPrint

可以在这里打印上自己的名字做个防伪认证:)

void InfoPrint()
{//打印提示信息SetPos(64, 15);printf("不能穿墙,不能咬到自己\n");SetPos(64, 16);printf("用 ↑. ↓. ←. → 分别控制蛇的移动.");SetPos(64, 17);printf("F1 为加速,F2 为减速\n");SetPos(64, 18);printf("ESC :退出游戏.space:暂停游戏.");SetPos(64, 20);printf("CSDN:fhvyxyci");
}

分数和每次吃食物的得分由于要刷新,就不在初始化的时候打印了。

5. 2. 5 SnakeInit

void SnakeInit(pSnake ps);

初始化蛇的步骤:

  1. 头插出一个有5个节点的链表
  2. 把这5个节点打印出来
  3. 初始化结构体其他数据
void SnakeInit(pSnake ps)
{pSnakeNode cur = NULL;for (int i = 0; i < 5; i++)	//初始设置长度为5{//创建一个节点,这一步也封装成 BuyNodepSnakeNode cur = (pSnakeNode)malloc(sizeof(SnakeNode));if (!cur){perror("SnakeInit()::malloc()");exit(1);}cur->x = X_INIT + 2 * i;	//X_INIT和Y_INIT是宏定义,方便修改初始坐标cur->y = Y_INIT;cur->next = NULL;//头插if (!ps->_Head){ps->_Head = cur;}else{cur->next = ps->_Head;ps->_Head = cur;}}//打印蛇cur = ps->_Head;while (cur){SetPos(cur->x, cur->y);wprintf(L"%lc", SNAKE_BODY);cur = cur->next;}//初始化数据ps->_Dir = RIGHT;		//初始方向为右ps->_FoodAdd = 10;		//初始食物的分数为10ps->_Score = 0;			//初始分数ps->_SleepTime = 200;	//_SleppTime与速度有关ps->_Sta = NORMAL;		//初始状态是正常
}

5. 2. 6 CreatFood

这个函数不是只在初始化的时候调用,写的时候可能要注意一下。

void CreatFood(pSnake ps)
{int x = 0, y = 0;
again:do{x = rand() % 55 + 2;	//注意范围y = rand() % 26 + 1;} while (x % 2 == 1);	//x必须是偶数pSnakeNode cur = ps->_Head;while (cur){//检查食物是否与身体重合if (cur->x == x && cur->y == y)goto again;		//当然,goto语句一般不推荐使用,你可以改造一下这里的逻辑,换成循环cur = cur->next;}//食物要有x,y坐标,那不如直接把它做成一个SnakeNode,这样还可以方便后面吃食物pSnakeNode food = (pSnakeNode)malloc(sizeof(SnakeNode));if(!food){perror("CreatFood()::malloc()");exit(1);}food->x = x;food->y = y;food->next = NULL;ps->_Food = food;//打印食物SetPos(ps->_Food->x, ps->_Food->y);wprintf(L"%c", FOOD);
}

剩下的逻辑在后面的博客中介绍。
谢谢你的阅读,喜欢的话来个点赞收藏评论关注吧!
我会持续更新更多优质文章

这篇关于大二必做项目贪吃蛇超详解之中篇游戏设计与分析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

不懂推荐算法也能设计推荐系统

本文以商业化应用推荐为例,告诉我们不懂推荐算法的产品,也能从产品侧出发, 设计出一款不错的推荐系统。 相信很多新手产品,看到算法二字,多是懵圈的。 什么排序算法、最短路径等都是相对传统的算法(注:传统是指科班出身的产品都会接触过)。但对于推荐算法,多数产品对着网上搜到的资源,都会无从下手。特别当某些推荐算法 和 “AI”扯上关系后,更是加大了理解的难度。 但,不了解推荐算法,就无法做推荐系

这15个Vue指令,让你的项目开发爽到爆

1. V-Hotkey 仓库地址: github.com/Dafrok/v-ho… Demo: 戳这里 https://dafrok.github.io/v-hotkey 安装: npm install --save v-hotkey 这个指令可以给组件绑定一个或多个快捷键。你想要通过按下 Escape 键后隐藏某个组件,按住 Control 和回车键再显示它吗?小菜一碟: <template

如何用Docker运行Django项目

本章教程,介绍如何用Docker创建一个Django,并运行能够访问。 一、拉取镜像 这里我们使用python3.11版本的docker镜像 docker pull python:3.11 二、运行容器 这里我们将容器内部的8080端口,映射到宿主机的80端口上。 docker run -itd --name python311 -p

性能分析之MySQL索引实战案例

文章目录 一、前言二、准备三、MySQL索引优化四、MySQL 索引知识回顾五、总结 一、前言 在上一讲性能工具之 JProfiler 简单登录案例分析实战中已经发现SQL没有建立索引问题,本文将一起从代码层去分析为什么没有建立索引? 开源ERP项目地址:https://gitee.com/jishenghua/JSH_ERP 二、准备 打开IDEA找到登录请求资源路径位置

OpenHarmony鸿蒙开发( Beta5.0)无感配网详解

1、简介 无感配网是指在设备联网过程中无需输入热点相关账号信息,即可快速实现设备配网,是一种兼顾高效性、可靠性和安全性的配网方式。 2、配网原理 2.1 通信原理 手机和智能设备之间的信息传递,利用特有的NAN协议实现。利用手机和智能设备之间的WiFi 感知订阅、发布能力,实现了数字管家应用和设备之间的发现。在完成设备间的认证和响应后,即可发送相关配网数据。同时还支持与常规Sof

在cscode中通过maven创建java项目

在cscode中创建java项目 可以通过博客完成maven的导入 建立maven项目 使用快捷键 Ctrl + Shift + P 建立一个 Maven 项目 1 Ctrl + Shift + P 打开输入框2 输入 "> java create"3 选择 maven4 选择 No Archetype5 输入 域名6 输入项目名称7 建立一个文件目录存放项目,文件名一般为项目名8 确定

6.1.数据结构-c/c++堆详解下篇(堆排序,TopK问题)

上篇:6.1.数据结构-c/c++模拟实现堆上篇(向下,上调整算法,建堆,增删数据)-CSDN博客 本章重点 1.使用堆来完成堆排序 2.使用堆解决TopK问题 目录 一.堆排序 1.1 思路 1.2 代码 1.3 简单测试 二.TopK问题 2.1 思路(求最小): 2.2 C语言代码(手写堆) 2.3 C++代码(使用优先级队列 priority_queue)

怎么让1台电脑共享给7人同时流畅设计

在当今的创意设计与数字内容生产领域,图形工作站以其强大的计算能力、专业的图形处理能力和稳定的系统性能,成为了众多设计师、动画师、视频编辑师等创意工作者的必备工具。 设计团队面临资源有限,比如只有一台高性能电脑时,如何高效地让七人同时流畅地进行设计工作,便成为了一个亟待解决的问题。 一、硬件升级与配置 1.高性能处理器(CPU):选择多核、高线程的处理器,例如Intel的至强系列或AMD的Ry

SWAP作物生长模型安装教程、数据制备、敏感性分析、气候变化影响、R模型敏感性分析与贝叶斯优化、Fortran源代码分析、气候数据降尺度与变化影响分析

查看原文>>>全流程SWAP农业模型数据制备、敏感性分析及气候变化影响实践技术应用 SWAP模型是由荷兰瓦赫宁根大学开发的先进农作物模型,它综合考虑了土壤-水分-大气以及植被间的相互作用;是一种描述作物生长过程的一种机理性作物生长模型。它不但运用Richard方程,使其能够精确的模拟土壤中水分的运动,而且耦合了WOFOST作物模型使作物的生长描述更为科学。 本文让更多的科研人员和农业工作者