大二必做项目贪吃蛇超详解之下篇游戏核心逻辑实现

news/2024/9/19 2:04:00/ 标签: 开发语言, c语言, 数据结构, visual studio

贪吃蛇系列文章

  1. 上篇win32库介绍
  2. 中篇设计与分析
  3. 下篇游戏主逻辑

可以在Gitee上获取贪吃蛇代码。

文章目录

  • 贪吃蛇系列文章
  • 5. 核心逻辑实现分析
    • 5. 3 GameRun
      • 5. 3. 1 PrintScore
      • 5. 3. 2 CheckVK
      • 5. 3. 3 BuyNewNode
      • 5. 3. 4 NextIsFood
      • 5. 3. 4 EatFood
      • 5. 3. 5 NotFood
      • 5. 3. 6 CheckIsWall和CheckIsSelf
    • 5. 4 GameOver
  • 6. 已知Bug与一些可能的改进意见


5. 核心逻辑实现分析

5. 3 GameRun

这个部分需要完成的任务:

  1. 游戏运行期间,右侧刷新分数
  2. 根据游戏状态检查游戏是否继续,如果是状态是NORMAL,游戏继续,否则游戏结束。
  3. 如果游戏继续,就是检测按键情况,确定蛇下一步的方向,或者是否加速减速,是否暂停或者退出游戏。
  4. 检查下一步会不会吃到食物,然后走一步

需要用到的虚拟按键值:

上: 	VK_UP
下: 	VK_DOWN
左: 	VK_LEFT
右: 	VK_RIGHT
空格:	VK_SPACE
ESC:	VK_ESCAPE
F1: 	VK_F1
F2: 	VK_F2

我们先来简单地分析一下如果没有死亡走一步大概是怎么走的:

  1. 通过前几步找到下一步走到的位置,并根据这个坐标创建一个SnakeNode作为走一步之后的头
  2. 如果下一步是食物,那就直接把这个新节点头插到蛇身上,然后再创建一个食物
  3. 如果下一步不是食物,不仅要把新节点头插上去,还需要遍历链表,把最后一个位置打印上空格(不然蛇就会越来越长),再把最后一个节点释放掉。

现在我们解释一下Snake结构体中的_SleepTime是怎么控制速度的。
首先我们要明确:程序的运行速度是非常快的,对于贪吃蛇这样的小项目来说,所有的代码都可以看作是瞬间完成的,如果直接执行,那贪吃蛇一定会在我们反应过来之前直接死亡,所以说我们需要使用Sleep函数让函数停下来一会儿来控制速度。

Sleep(unsigned long);

参数的单位是毫秒,当程序运行到这里时,可以让程序暂停参数的时长
我们就可以借助这个函数来控制贪吃蛇的速度了,在每次走一步之后,Sleep(ps->_SleepTime);就可以了。

那么我们就可以写出来

void GameRun(pSnake ps)
{do	//这个循环用来控制一场游戏何时结束{//打印分数//打印分数应该放在最前面,不然会导致贪吃蛇在走出第一步的时候右边还没有分数PrintScore(ps);//检查按键CheckVK(ps);//新建一个节点,作为下一个头pSnakeNode nextnode = BuyNewNode(ps);//检查下一个是不是食物,如果是,加分并连接,走一步if (NextIsFood(nextnode,ps))EatFood(nextnode,ps);elseNotFood(nextnode,ps);//检查是否撞到自己CheckIsSelf(ps);//检查是否撞墙CheckIsWall(ps);//休眠,控制速度Sleep(ps->_SleepTime);} while (ps->_Sta == NORMAL);
}

5. 3. 1 PrintScore

void PrintScore(pSnake ps)
{SetPos(64, 10);printf("得分:%d ", ps->_Score);printf("每个食物得分:%2d分", ps->_FoodAdd);
}

在打印每个食物得分的时候,因为可能这个数字是一位数,所以要使用%2d来使打印出来的数字占两个位置,不然可能会出现这样的情况:

每个食物得分:12//减速
每个食物得分:82

之前打印上的 2 如果不被覆盖掉的话是不会消失的。

5. 3. 2 CheckVK

这里用到了我们自定义的一个宏:KEY_PRESS

#define KEY_PRESS(VK) ( (GetAsyncKeyState(VK) & 0x1) ? 1 : 0 )

不了解的话可以前往翻看这篇博客:win32库介绍。
要实现贪吃蛇转弯,但是显而易见地转弯的时候不能转到自己的身后,所以要加以限制
另外加速和减速时,当然不能让玩家无限制地加速或减速下去,必须做出一定限制

void CheckVK(pSnake ps)
{if (KEY_PRESS(VK_UP))			//上{if (ps->_Dir != DOWN)ps->_Dir = UP;}else if (KEY_PRESS(VK_DOWN))	//下{if (ps->_Dir != UP)ps->_Dir = DOWN;}else if (KEY_PRESS(VK_LEFT))	//左{if (ps->_Dir != RIGHT)ps->_Dir = LEFT;}else if (KEY_PRESS(VK_RIGHT))	//右{if (ps->_Dir != LEFT)ps->_Dir = RIGHT;}else if (KEY_PRESS(VK_F1))		//加速{if (ps->_SleepTime >= 50)	//速度上限{ps->_SleepTime -= 30;ps->_FoodAdd += 2;		//记得更改每个食物的分数}}else if (KEY_PRESS(VK_F2))		//减速{if (ps->_SleepTime < 320)	//速度下限{ps->_SleepTime += 30;ps->_FoodAdd -= 2;}}else if (KEY_PRESS(VK_SPACE))	//空格,暂停{while (!KEY_PRESS(VK_SPACE)){//在再次点击空格之前,循环休眠Sleep(200);}}else if (KEY_PRESS(VK_ESCAPE))	//ESC,主动退出{ps->_Sta = ESC;				//注意这个ESC是我们在上篇博客中写的枚举类型的成员}
}

5. 3. 3 BuyNewNode

一个申请新的链表节点的函数,只是它存储的数据需要通过计算。
不过要注意,由于我们是用了宽字符,涉及到X方向的坐标改变时,我们需要±2,而Y方向还是1

pSnakeNode BuyNewNode(pSnake ps)
{int newx = ps->_Head->x;int newy = ps->_Head->y;//根据链表的头结点的位置和蛇的方向找的下一个位置的头的位置if (ps->_Dir == LEFT)newx -= 2;else if (ps->_Dir == RIGHT)newx += 2;else if (ps->_Dir == UP)newy -= 1;else if (ps->_Dir == DOWN)newy += 1;pSnakeNode nextnode = (pSnakeNode)malloc(sizeof(SnakeNode));if (!nextnode){perror("NoFood()::malloc()");exit(1);}nextnode->x = newx;nextnode->y = newy;return nextnode;
}

5. 3. 4 NextIsFood

检查下一步是不是食物,只需要将新的头(还未插入)的数据和食物的坐标进行比较就可以了。

bool NextIsFood(pSnakeNode nextnode, pSnake ps)
{if (ps->_Food->x == nextnode->x && ps->_Food->y == nextnode->y)return true;elsereturn false;
}

5. 3. 4 EatFood

吃下食物有这么几个步骤:

  1. newnode头插到蛇身上去
  2. 删除食物并重新生成一个(注意食物节点是动态开辟的,要free掉)
  3. 把原来的食物的图标用蛇身覆盖掉
  4. 加分
void EatFood(pSnakeNode newhead, pSnake ps)
{//将newhead头插到蛇身上newhead->next = ps->_Head;ps->_Head = newhead;//删除食物free(ps->_Food);ps->_Food = NULL;CreatFood(ps);//打印蛇头把食物覆盖掉SetPos(ps->_Head->x, ps->_Head->y);wprintf(L"%c", SNAKE_BODY);//也可以直接刷新整个蛇身,显示效果可能稍有差异//pSnakeNode cur = ps->_Head;//while (cur)//{//	SetPos(cur->x, cur->y);//	wprintf(L"%c", SNAKE_BODY);//	cur = cur->next;//}//加分ps->_Score += ps->_FoodAdd;
}

5. 3. 5 NotFood

如果下一步不是食物有这么几个步骤:

  1. 把新节点头插上去
  2. 打印新的头节点
  3. 把原来的尾节点打印的符号用空格覆盖掉
  4. 尾删
void NotFood(pSnakeNode newhead, pSnake ps)
{//头插newhead->next = ps->_Head;ps->_Head = newhead;//打印新头SetPos(ps->_Head->x, ps->_Head->y);wprintf(L"%c", SNAKE_BODY);//将尾节点的符号用空格顶替掉pSnakeNode cur = ps->_Head;while (cur->next->next)	//这个循环最终会找到尾节点的上一个节点cur = cur->next;SetPos(cur->next->x, cur->next->y);printf("  ");//尾删free(cur->next);cur->next = NULL;
}

5. 3. 6 CheckIsWall和CheckIsSelf

这两个函数就是死亡判定了。
检测是否撞墙,只需要判断蛇身是否出界就可以了。
检测是否撞到自己,就需要**遍历链表来一一对比 **了。

void CheckIsSelf(pSnake ps)
{pSnakeNode cur = ps->_Head->next;//遍历检测是否撞到自己while (cur){if (cur->x == ps->_Head->x && cur->y == ps->_Head->y){ps->_Sta = KILL_BY_SELF;	//撞到了就更改状态break;}cur = cur->next;}
}void CheckIsWall(pSnake ps)
{//检测头节点的坐标是否超出范围if (ps->_Head->x <= 0 || ps->_Head->x >= 58 || ps->_Head->y <= 0 || ps->_Head->y >= 27)ps->_Sta = KILL_BY_WALL;
}

那么至此,GameRun函数就写完了,游戏已经能基本正常的运行起来了。

5. 4 GameOver

那么剩下的便是收尾工作了,这个游戏中使用了动态内存管理,在不在进行使用之后,必须进行释放,不然会导致内存泄漏。
这个函数要完成以下内容:

  1. 打印死亡信息,告诉玩家是怎么死亡的(当然,也可以方便调试)
  2. 回收内存

打印死亡信息只需要根据ps->_Sta的不同状态设置不同的语句就可以了。
而蛇的销毁就是链表的销毁,也不赘述了。

void GameOver(pSnake ps)
{//打印死亡信息(用于调试)SetPos(15, 14);if (ps->_Sta == KILL_BY_SELF)printf("你撞到了自己");else if (ps->_Sta == KILL_BY_WALL)printf("你撞墙了");elseprintf("正常退出");//释放蛇的内存pSnakeNode cur = ps->_Head;while (cur){pSnakeNode next = cur->next;free(cur);cur = next;}free(ps->_Food);ps->_Food = NULL;
}

那么接下来就是回到上篇博客的游戏主逻辑中,开始询问玩家要不要再来一把了。

6. 已知Bug与一些可能的改进意见

我们先来看上篇中的这个循环:

while (_kbhit())	//_kbhit()检测是否有按键被按下
{//使用 _getch() 获取按下的键_getch();
}

处理的Bug是如果在第二次及以后的游戏(也就是输入了Y进行了再来一把)中,如果使用了F1加速,就会在本把游戏结束时成为这样:
2
这个Y并不是手动打上去的,而是由于其他原因上去的,并且这个Y还可以再被getchar()读取下来,导致游戏再无法退出。
至于成因,可以看一眼:
在 cmd 中先输入Y,回车,F1,回车,就会看到这样的情况:
2
而上面的代码就可以解决这个问题了。

一些可能实现的改进:

  1. 多个食物
  2. 地图大小可自定义
  3. 增加游戏时间显示
  4. 增加胜利判断(蛇身占满整个地图)

贪吃蛇代码可以在Gitee上获取,喜欢的话点个star吧。
谢谢你的阅读,喜欢的话来个点赞收藏评论关注吧!
我会持续更新更多优质文章


http://www.ppmy.cn/news/1520309.html

相关文章

STM32F401使用float浮点运算崩溃的一个解决实例

今天使用STM32F401开发大彩的串口屏通信&#xff0c;串口使用USART1,DMA通信&#xff0c;系统是FreeRTOS。 使用大彩提供的hmi_driver&#xff0c;执行到SetTextFloat这个函数时崩溃 该函数原型&#xff1a; void SetTextFloat(uint16 screen_id,uint16 control_id,float va…

dinput8.dll错误应该如何修复呢?五种快速修复dinput8.dll错误的问题

dinput8.dll文件是DirectInput库的一部分&#xff0c;主要负责处理游戏控制器的输入&#xff0c;如键盘、鼠标和游戏手柄等。这个文件通常位于Windows系统的System32文件夹中&#xff0c;是许多游戏和应用程序正常运行所必需的组件。它通过提供一个统一的接口来管理不同类型的输…

持续集成与持续部署(CI/CD)的深入探讨

在现代软件开发中&#xff0c;持续集成&#xff08;CI&#xff09;和持续部署&#xff08;CD&#xff09;已成为不可或缺的实践。这些方法旨在加快软件交付的速度&#xff0c;同时提高软件的质量和稳定性。通过CI/CD&#xff0c;开发团队可以频繁地将代码更改集成到主分支&…

集成电路学习:什么是ARM先进精简指令集计算机

ARM&#xff1a;先进精简指令集计算机 ARM先进精简指令集计算机&#xff08;Advanced RISC Machine&#xff0c;简称ARM&#xff09;是一种基于精简指令集计算机&#xff08;RISC&#xff09;原则的计算机处理器架构&#xff0c;由英国的ARM公司开发。这种架构以其低功耗和高性…

C++STL~~list

文章目录 一、list的概念二、list的使用三、list的练习四、与vector的对比五、总结 一、list的概念 list 是一种容器&#xff0c;实现了双向链表结构 它具有以下特点&#xff1a; 动态大小&#xff0c;可按需增减元素数量。高效的插入和删除操作&#xff0c;在任意位置插入和…

抽象和接口

a.抽象&#xff08;abstract&#xff09; 1. 定义 a. 抽象类&#xff1a;在普通类里增加了抽象方法。 b. 抽象方法&#xff1a;没有具体的执行方法&#xff0c;没有方法体的方法。 2. 总结 a. 因为抽象方法没有方法体&#xff0c;无法执行&#xff0c;所以不能…

Hackme靶机通关攻略

1.首先注册用户&#xff0c;登录 2.登录后&#xff0c;显示让我们查找自己喜欢的书&#xff0c;我们直接单击search&#xff0c;会列出很多书 3.随便选择一本书进行查询&#xff0c;与此同时进行抓包 4.放到重放器中&#xff0c;将数据改为1*&#xff0c;将数据包另存为1.txt&a…

c++11的学习

1.初始化列表 在C98中&#xff0c;标准允许使用花括号{}对数组或者结构体元素进行统一的列表初始值设定。 struct Fun {int x;int y; }; struct Date {Date(int _year, int _month, int _day):year( _year),month(_month),day(_day){}int year 2005;int month 01;int day …

Mac装机必备软件有哪些?苹果电脑实用软件推荐

刚刚入手MacBook或者苹果电脑需要安装哪些软件呢&#xff1f;越来越多的人使用 Mac&#xff0c;各种功能、各式各样的 Mac 软件也是五花八门。刚拿到 Mac 的小伙伴们可能会有点迷茫&#xff0c;今天就帮大家分类整理一些装机必备好用的 App&#xff0c;保证个个是神器&#xff…

区间的合并

区间合并的说明 业务中的区间合并是比较常见的需求&#xff0c;区间合并的核心有两点&#xff1a; 合并前排序&#xff0c;后面处理起来可以简单很多&#xff1b;两个区间合并&#xff0c;这是多个区间合并的基础。 完整代码 区间类 /*** 区间类&#xff08;left、right可…

udp可靠传输中ACK与NACK的选择

文章目录 介绍ACKNACK丢包触发常见问题包序号比较RTT&RTO估算乱序包UDP包乱序的原因乱序现象所说明的问题 乱序处理ACK与NACK结合使用数据包结构发送方策略接收方策略动态调整 介绍 在 UDP 可靠传输中&#xff0c;ACK&#xff08;Acknowledgment&#xff09;和 NACK&#x…

前端页面实现面料的模拟

实现面料模拟一直是一个比较难的问题&#xff0c;因此今天说一下我的具体实现方案&#xff0c;和代码 先展示结果&#xff0c;这是没有加物理效果的衣服 这是加了物理效果的衣服 是不是效果很明显 首先&#xff0c;先说原理&#xff0c;通过修改每个顶点的位置来实现衣服的移…

数据恢复工具,电脑+手机双端,十分好用!

哈喽&#xff0c;各位小伙伴们好&#xff0c;我是给大家带来各类黑科技与前沿资讯的小武。 今天给大家安利两款数据恢复工具&#xff0c;分别为电脑手机双端&#xff0c;无论是因为格式化误操作、设备损坏还是其他意外情况&#xff0c;都能轻松找回重要的文件、照片、视频等数…

【unity实战】使用新版输入系统Input System+Rigidbody实现第三人称人物控制器(附项目源码)

最终效果 前言 使用CharacterController实现3d角色控制器&#xff0c;之前已经做过很多了&#xff1a; 【unity小技巧】unity最完美的CharacterController 3d角色控制器&#xff0c;实现移动、跳跃、下蹲、奔跑、上下坡、物理碰撞效果&#xff0c;复制粘贴即用 【unity实战】C…

基于大数据的电信诈骗行为可视化系统含预测研究【lightGBM,XGBoost,随机森林】

文章目录 有需要本项目的代码或文档以及全部资源&#xff0c;或者部署调试可以私信博主项目介绍 电信诈骗预测与分析系统项目概述系统架构详细功能描述1. 数据预处理2. 数据可视化与分析3. 机器学习预测4. 系统集成与用户界面 技术亮点应用价值未来展望lightGBMXGBoost随机森林…

【滑动窗口】将 x 减到 0 的最小操作数

将 x 减到 0 的最小操作数 将 x 减到 0 的最小操作数题目思路讲解代码书写 将 x 减到 0 的最小操作数 题目 题目链接: 将 x 减到 0 的最小操作数 思路讲解 按照题目的思路去做这一题是非常恶心的, 因此我们采用正难则反思路. 将问题转换为: 求中间某一个最长的数组长度, 使…

监控Nginx负载均衡后端服务器状态的策略与实践

在Nginx负载均衡的部署中&#xff0c;监控后端服务器的状态对于确保高可用性和服务连续性至关重要。通过检测后端服务器的状态&#xff0c;可以及时发现问题并采取措施&#xff0c;如故障转移或服务重启。本文将详细介绍如何检测Nginx负载均衡后端服务器的状态&#xff0c;包括…

PHP之 ThinkPHP5配置redis缓存

tp config.php cache > [// 使用复合缓存类型type > complex,// 默认使用的缓存default > [// 驱动方式type > File,// 缓存保存目录path > CACHE_PATH,// 缓存前缀prefix > ,// 缓存有效期 0表示永久缓存expire > 0,],// 文件缓存file >…

Prometheus+Grafana监控数据可视化

上一篇文章讲了prometheus的简单使用&#xff0c;这一篇就先跳过中间略显枯燥的内容&#xff0c;来到监控数据可视化。 一方面&#xff0c;可视化的界面看着更带劲&#xff0c;另一方面&#xff0c;也更方便我们直观的查看监控数据&#xff0c;方便后面的学习。 Grafana安装与…

RDP最小化之后仍然保持UI渲染的方法

RDP最小化之后仍然保持UI渲染的方法 1、运行regedit 2、找到注册表项HKEY_CURRENT_USER\Software\Microsoft\TerminalServer Client 3、新建一个类型为DWORD的注册表项RemoteDesktop_SuppressWhenMinimized并设置值为2 4、然后找到注册表项HKEY_CURRENT_USER\Software\Wow6432N…