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

news2024/11/15 10:49:44

贪吃蛇系列文章

  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);
		else
			NotFood(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;
	else
		return 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("你撞墙了");
	else
		printf("正常退出");
	//释放蛇的内存
	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.coloradmin.cn/o/2101664.html

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

相关文章

【OpenLayers 进阶】添加滤镜改变底图样式

目录 一、前言二、准备工作三、实现方式四、总结 一、前言 项目实施过程中&#xff0c;需要根据不同的业务场景需求变换地图样式。如果客户提供的底图服务或自建底图服务是类似Mapbox这种矢量切片&#xff0c;那只要按照需求配置不同的样式文件即可。如果没有矢量切片&#xff…

浅谈人工智能之Windows:基于ollama进行本地化大模型部署

浅谈人工智能之Windows&#xff1a;基于ollama进行本地化大模型部署 引言 随着人工智能技术的飞速发展&#xff0c;大型语言模型&#xff08;LLMs&#xff09;已经成为推动自然语言处理领域进步的关键力量。然而&#xff0c;传统的云部署方式可能带来数据隐私、成本以及访问速…

pikachu文件包含漏洞靶场通关攻略

本地文件包含 先上传一个jpg文件&#xff0c;内容写上<?php phpinfo();?> 上传成功并且知晓了文件的路径 返回本地上传&#xff0c;并../返回上级目录 可以看到我们的php语句已经生效 远程文件包含 在云服务器上创建一个php文件 然后打开pikachu的远程文件包含靶场&…

鸿蒙开发培训多少钱?

随着物联网技术的发展&#xff0c;HarmonyOS(鸿蒙系统)作为华为推出的一款面向全场景的分布式操作系统&#xff0c;正在逐步成为开发者们关注的焦点。对于那些想要进入这个新兴领域的开发者来说&#xff0c;参加鸿蒙开发培训是一个不错的选择。那么&#xff0c;这样的培训究竟需…

【Hadoop|HDFS篇】HDFS概述

1. HDFS产出背景及定义 1.1 HDFS产生背景 随着数据量越来越大&#xff0c;在一个操作系统存不下所有的数据&#xff0c;那么就分配到更多的操作系 统管理的磁盘中&#xff0c;但是不方便管理和维护&#xff0c;迫切需要一种系统来管理多台机器上的文件&#xff0c;这 就是分布…

Oracle授权如何购买?多少钱?如何计算?

前言 作为DBA时常也会遇到一些商务的问题&#xff0c;比如购买Oracle 的授权&#xff0c;比如老板问用oracle有没有法律风险&#xff0c;这个组件是否收费&#xff1f;如何计算授权数&#xff1f;等等&#xff0c;本文根据博主的经验和一些Oracle公开的资料&#xff0c;来做一个…

DBNET文字检测

原文:DBNET文字检测 - 知乎 (zhihu.com) 一、DBNET介绍 DBNET核心采用的是基于分割的做法进行文本检测,即将每个文本块都进行语义分割,然后对分割概率图进行简单二值化、最终转化得为box或者poly格式的检测结果。除去网络设计方面的差异,最大特点是引入了Differentiable …

Python 从入门到实战6(二维列表)

我们的目标是&#xff1a;通过这一套资料学习下来&#xff0c;通过熟练掌握python基础&#xff0c;然后结合经典实例、实践相结合&#xff0c;使我们完全掌握python&#xff0c;并做到独立完成项目开发的能力。 之前的文章我们通过举例学习了python 中列表的相关操作&#xff0…

什么是阿凡达2.0直播模式?

要了解什么是什么是阿凡达2.0直播模式,首先要了解什么是的阿凡达直播模式。 我们知道真人直播&#xff0c;播不了几个小时&#xff0c;主播就讲累了。且真人主播的价格又贵&#xff0c;以小时计费。所以很多数字人厂商推出了数字人直播。用数字人代替真人直播。在前几年的时候…

强烈建议!重罚自燃车企

文 | AUTO芯球 作者 | 雷慢 想想就叫人害怕啊&#xff0c; 广东惠州电车自燃那个事&#xff0c;你们看了吗 3辆汽车和多辆电动自行车被烧毁&#xff0c;住宅变危房&#xff0c; 最触目惊心的还是浓烟&#xff0c;许多住户咳呛不止&#xff0c; 周边小区也惨遭浓烟毒害&…

Css:css的属性选择器vs关系选择器及css中伪元素

css的属性选择器&#xff1a; 注&#xff1a;属性值只能由数字&#xff0c;字母&#xff0c;下划线&#xff0c;中划线组成&#xff0c;并且不能以数字开头。 1、[属性] 选择含有指定属性的元素&#xff0c;用[]中括号表示。 <style> [title]{color:red;} p[title]{col…

「Python数据分析」Pandas进阶,使用groupby分组聚合数据(一)

在数据分析过程中&#xff0c;groupby语句&#xff0c;起到对原始数据集&#xff0c;进行分组和聚合的作用。我们在进行数据处理的时候&#xff0c;经常需要对不同的数据维度&#xff0c;以及不同的数据切片集合&#xff0c;进行操作和处理。 比如说&#xff0c;假设我们有全国…

零基础国产GD32单片机编程入门(十二)FreeRTOS实时操作系统移植含源码

文章目录 一.概要二.什么是实时操作系统三.FreeRTOS的特性四.FreeRTOS的任务详解1.任务函数定义2.任务的创建3.任务的调度原理 五.FreeRTOS系统移植到GD32F103C8T6单片机1.硬件准备2.程序移植3.调试FreeRTOS任务调度 六.工程源代码下载七.小结 一.概要 FreeRTOS是一个迷你的实…

海外直播对网速、带宽、安全的要求

要满足海外直播的要求&#xff0c;需要拥有合适的网络配置。在全球化的浪潮下&#xff0c;海外直播正逐渐成为企业、个人和各类组织的重要工具。不论是用于市场推广、品牌宣传&#xff0c;还是与观众互动&#xff0c;海外直播都为参与者带来了丰富的机会。然而&#xff0c;确保…

KTH2502 系列车规数字锁存霍尔效应传感器

KTH2502 采用了先进的斩波技术&#xff0c;集成了温度补偿电路和过流、负压保护电路&#xff0c;具有卓越的灵敏度和温度稳定 性。磁场通过数字双极锁存器输出指示。该芯片具有开漏输出级以及 30mA 的灌电流能力。 2.7~32 V 的宽电压工 作范围&#xff0c;反极性保护高达 -22…

渗透测试靶机--- DC系列 DC-9

渗透测试靶机— DC系列 DC-9 开启靶机&#xff0c;依旧登录窗&#xff0c;平平无奇 扫描信息 访问页面看看 页面中存在查询接口&#xff0c;这里页面输入的内容是get请求&#xff0c;可以尝试sql注入&#xff0c;如果是post请求&#xff0c;那么抓包。将请求体放入sqlmap跑一下…

太速科技-基于Kintex-7 XC7K160T 的CameraLink转四路光纤数据转发卡(Full Camera Link图像转万兆以太网适配器 )

基于Kintex-7 XC7K160T 的CameraLink转四路光纤数据转发卡&#xff08;Full Camera Link图像转万兆以太网适配器 &#xff09; 一、板卡概述 该板卡是一款CameraLink&#xff08;Full&#xff09;转4路光纤接口板&#xff0c;可以实现1路CamerLink Full模式的图像信号转换成…

Druid未授权访问解决

Druid未授权访问原因分析 漏洞说明&#xff1a;Druid由阿里巴巴数据库出品&#xff0c;为监控而生的数据库连接池&#xff0c;并且Druid可以提供监控&#xff0c;监控SQL的执行时间、监控Web URI的请求、Session监控等功能&#xff0c;使用广泛。 需要明确&#xff1a; Druid…

物联网(IoT)支持的小型水处理厂实时硬件在环(HIL)仿真

这篇论文的标题是《Real-Time Hardware-In-The-Loop Simulation of IoT-Enabled Mini Water Treatment Plant》&#xff0c;作者是 Mohamad Taib Miskon 等人&#xff0c;发表在 2024 年 IEEE 自动控制与智能系统国际会议&#xff08;I2CACIS&#xff09;上。以下是该论文的主要…

搭建 xxl-job 调度中心

文章目录 1、初始化“调度数据库”2、修改“调度中心”配置3、打包运行”调度中心“ 1、初始化“调度数据库” 请下载项目源码并解压&#xff0c;获取 “调度数据库初始化SQL脚本” 并执行即可。 “调度数据库初始化SQL脚本” 位置为&#xff1a;/xxl-job/doc/db/tables_xxl_j…