GitHub经典贪吃蛇思路解析

2024-08-24 07:36

本文主要是介绍GitHub经典贪吃蛇思路解析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

贪吃蛇游戏文档

游戏简介

贪吃蛇是一款经典的控制台游戏,玩家需要操控一条蛇去吃掉随机出现在屏幕上的食物来增长蛇的长度,并避免蛇头撞到墙壁或自己的身体。本游戏使用 C++ 编写,利用 Windows API 实现了控制台窗口内的动态游戏界面。

开发环境
  • 操作系统:Windows
  • 编程语言:C++
  • 编译器:适用于 C++ 的任何兼容编译器(如 MSVC)
  • 依赖库<iostream><list><thread><chrono><windows.h><conio.h>
游戏启动
  1. 运行环境设置:确保你的开发环境已正确配置好 Windows API 和 C++ 编译器。
  2. 编译游戏源码:使用支持 C++ 的编译器编译游戏源代码。
  3. 启动游戏:通过命令行运行编译后的可执行文件。
游戏界面
  • 屏幕尺寸:游戏默认屏幕尺寸为 120x30,但会根据实际控制台窗口大小自动调整。
  • 初始布局:游戏开始时,屏幕上会显示一个由等号 = 构成的边界框,并在顶部中间位置显示游戏标题和分数。
void GetConsoleSize(int& width, int& height) {CONSOLE_SCREEN_BUFFER_INFO csbi;GetConsoleScreenBufferInfo(GetStdHandle(STD_OUTPUT_HANDLE), &csbi);width = csbi.srWindow.Right - csbi.srWindow.Left + 1;height = csbi.srWindow.Bottom - csbi.srWindow.Top + 1;
}int main() {HANDLE hConsole = CreateConsoleScreenBuffer(GENERIC_READ | GENERIC_WRITE, 0, NULL, CONSOLE_TEXTMODE_BUFFER, NULL);SetConsoleActiveScreenBuffer(hConsole);DWORD dwBytesWritten = 0;while (true) {GetConsoleSize(nScreenWidth, nScreenHeight);// Create display screenwchar_t* screen = new wchar_t[nScreenHeight * nScreenWidth];for (int i = 0; i < nScreenHeight * nScreenWidth; i++) {screen[i] = L' ';}
游戏规则
  1. 移动控制
    • 使用键盘上的左右箭头键来控制蛇的移动方向。
    • 按下箭头键后,蛇头会在下一个时间间隔内按照指定方向移动一格。
    • 每次只能改变到相邻的方向(即不能立即从向左变为向右)。

这段代码是用于处理贪吃蛇游戏中的帧定时以及用户输入的部分。以下是逐行解析:

// Frame Timing, compensate for aspect ratio of command line
auto t1 = chrono::system_clock::now();
  • 这一行记录了当前的时间点 t1,作为后续计算时间差的基础。这是为了实现帧定时,即控制游戏的更新频率。
while ((chrono::system_clock::now() - t1) < ((nSnakeDirection % 2 == 1) ? 120ms : 200ms))
{
  • 这个 while 循环将持续执行,直到当前时间与记录的时间点 t1 之间的差达到了一个特定的毫秒数。具体来说,如果 nSnakeDirection 是奇数,等待时间为 120 毫秒;如果是偶数,等待时间为 200 毫秒。这样做是为了控制游戏的帧率,使游戏运行更加平滑,并且可以根据方向的不同调整速度。
    // Get Input,bKeyRight = (0x8000 & GetAsyncKeyState((unsigned char)('\x27'))) != 0;bKeyLeft = (0x8000 & GetAsyncKeyState((unsigned char)('\x25'))) != 0;
  • 这两行代码检测用户是否按下了右箭头键(\x27)或左箭头键(\x25)。GetAsyncKeyState 函数用于获取按键的状态,返回一个 16 位的整数,其中最高位(第 15 位)表示键是否被按下。通过与运算 (0x8000 & ...) 可以判断该位是否为 1,从而确定键是否被按下。
    if (bKeyRight && !bKeyRightOld){nSnakeDirection++;if (nSnakeDirection == 4) nSnakeDirection = 0;}if (bKeyLeft && !bKeyLeftOld){nSnakeDirection--;if (nSnakeDirection == -1) nSnakeDirection = 3;}
  • 这两个 if 语句检查是否检测到了新的按键动作。如果右键被按下并且之前没有被按下(即 bKeyRightOldfalse),则增加 nSnakeDirection 的值,并确保其不超过 3(即循环到 0)。同理,如果左键被按下并且之前没有被按下,则减少 nSnakeDirection 的值,并确保其不低于 0(即循环到 3)。这里的 nSnakeDirection 值通常表示蛇的四个可能的方向:0(向上)、1(向右)、2(向下)、3(向左)。
    bKeyRightOld = bKeyRight;bKeyLeftOld = bKeyLeft;
}
  • 最后这两行更新旧按键状态变量 bKeyRightOldbKeyLeftOld,以便下次循环时可以比较新的按键状态。

总结:这段代码的主要作用是在一个固定的时间间隔内处理用户的输入,并根据输入更新蛇的移动方向。同时,通过控制时间间隔,实现了游戏帧率的控制,确保游戏运行流畅且响应及时。

在这里插入代码片
 auto t1 = chrono::system_clock::now();while ((chrono::system_clock::now() - t1) < ((nSnakeDirection % 2 == 1) ? 120ms : 200ms)) {bKeyRight = (0x8000 & GetAsyncKeyState((unsigned char)('\x27'))) != 0;bKeyLeft = (0x8000 & GetAsyncKeyState((unsigned char)('\x25'))) != 0;if (bKeyRight && !bKeyRightOld) {nSnakeDirection++;if (nSnakeDirection == 4) nSnakeDirection = 0;}if (bKeyLeft && !bKeyLeftOld) {nSnakeDirection--;if (nSnakeDirection == -1) nSnakeDirection = 3;}bKeyRightOld = bKeyRight;bKeyLeftOld = bKeyLeft;}switch (nSnakeDirection) {case 0: snake1.push_front({ snake1.front().x, snake1.front().y - 1 }); break;case 1: snake1.push_front({ snake1.front().x + 1, snake1.front().y }); break;case 2: snake1.push_front({ snake1.front().x, snake1.front().y + 1 }); break;case 3: snake1.push_front({ snake1.front().x - 1, snake1.front().y }); break;}
  1. 食物生成
    • 食物随机生成在屏幕上的任意一个空闲位置。
    • 食物用 % 符号表示。
 do {nFoodX = rand() % nScreenWidth;nFoodY = (rand() % (nScreenHeight - 3)) + 3;} while (screen[nFoodY * nScreenWidth + nFoodX] != L' ');
  1. 得分机制
    • 当蛇头与食物重合时,蛇的长度会增加,同时得分增加。
    • 每吃掉一个食物,蛇会增长5个单位。
// Check collision with food
if (snake1.front().x == nFoodX && snake1.front().y == nFoodY) {nScore++;do {nFoodX = rand() % nScreenWidth;nFoodY = (rand() % (nScreenHeight - 3)) + 3;} while (screen[nFoodY * nScreenWidth + nFoodX] != L' ');for (int i = 0; i < 5; i++)snake1.push_back({ snake1.back().x, snake1.back().y });
}
  1. 碰撞检测
    • 若蛇头撞到墙壁或蛇身,则游戏结束。
    • 游戏结束时,屏幕上会出现提示信息,告知玩家按空格键重新开始游戏。
    • 检测与墙壁的碰撞 检测与食物的碰撞 检测与自身的碰撞
// Check collision with food
if (snake1.front().x == nFoodX && snake1.front().y == nFoodY) {nScore++;do {nFoodX = rand() % nScreenWidth;nFoodY = (rand() % (nScreenHeight - 3)) + 3;} while (screen[nFoodY * nScreenWidth + nFoodX] != L' ');for (int i = 0; i < 5; i++)snake1.push_back({ snake1.back().x, snake1.back().y });
}// Check collision with walls
if (snake1.front().x < 0 || snake1.front().x >= nScreenWidth ||snake1.front().y < 3 || snake1.front().y >= nScreenHeight) {bDead = true;
}// Check collision with self
for (list<snake>::iterator i = next(snake1.begin()); i != snake1.end(); ++i)if (i->x == snake1.front().x && i->y == snake1.front().y)bDead = true;
  1. 游戏结束与重新开始
    • 玩家可以通过按空格键来重新开始游戏。
 // Wait for space to restartwhile ((0x8000 & GetAsyncKeyState((unsigned char)('\x20'))) == 0);
游戏流程
  1. 初始化阶段

    • 游戏开始时,初始化屏幕大小、蛇的位置以及食物的位置。
    • 显示初始的游戏界面,包括边界、初始蛇形、食物位置以及得分。
  2. 主游戏循环

    • 在主循环中,不断检查键盘输入,更新蛇的移动方向。
    • 每过一定的时间间隔(根据方向不同,时间间隔也不同),蛇向前移动一格。
    • 检查蛇头是否吃到食物,如果是,则增加蛇的长度并更新得分。
    • 检查是否有碰撞发生,如果发生碰撞,则标记游戏结束。
    • 清除屏幕,重新绘制所有元素,包括蛇、食物和边界。
    • 如果游戏结束,显示游戏结束的信息。

清除屏幕并重新绘制所有游戏元素这一过程对于保持游戏的流畅性和视觉效果至关重要,它实际上就是维持游戏帧率的一部分工作。这个过程可以理解为游戏开发中的“帧循环”,每次循环都会执行以下步骤:

  1. 检测输入:检查用户的键盘或其他输入设备,以确定蛇的移动方向。
  2. 更新状态:根据输入更新游戏的状态,例如蛇的位置、食物的位置等。
  3. 碰撞检测:检查蛇是否撞到了自己或者边界,以及是否吃到了食物。
  4. 清理画布:清除上一帧的画面,为新的画面做准备。这是为了避免旧的图形残留,造成视觉混乱。
  5. 重新绘制:在屏幕上重新绘制所有的游戏元素,包括蛇的身体、食物以及游戏边界等。
  6. 检查游戏结束条件:如果游戏结束(如蛇撞到自身或墙壁),则显示游戏结束的消息。

这个循环通常会以固定的频率运行,即帧率(FPS,Frames Per
Second)。帧率越高,游戏看起来就越流畅。在贪吃蛇游戏中,这个频率不需要非常高,因为游戏本身并不需要极其复杂的动画效果。但是,保持一定的帧率对于用户体验来说仍然是重要的,因为它决定了游戏反应的速度和流畅度。

通过不断地重复这个过程,游戏能够呈现出动态的效果,让玩家感觉游戏是在连续地进行而不是断断续续的。因此,每次循环中清除屏幕然后重新绘制所有元素,是为了确保每一帧都是最新的状态,从而达到刷新帧率的效果。

  1. 游戏结束处理
    • 当检测到游戏结束条件满足时,停止蛇的移动。
    • 在屏幕上显示游戏结束的信息,并等待玩家按键。
    • 如果玩家按下了空格键,则重新开始游戏。
技术细节
  • 控制台操作:使用 GetConsoleScreenBufferInfoSetConsoleScreenBufferInfo 来获取和设置控制台的信息。
  • 异步按键检测:使用 GetAsyncKeyState 来实时检测键盘按键的状态。
  • 定时器:使用 <chrono> 库来实现游戏的定时功能,保证蛇的移动频率。
  • 动态内存管理:使用 newdelete[] 来分配和释放屏幕缓冲区。

通过以上文档,你应该能够清楚地了解这款游戏的工作原理以及如何开始游戏。

完整源码+注释

/*C++ 精品贪吃蛇*/
// 项目标题:C++精品贪吃蛇游戏#include <iostream>         // 包含用于输入输出的基本流库
#include <list>             // 包含标准库中的list容器
#include <thread>           // 包含线程库,用于延时
#include <chrono>           // 包含时间库,用于获取当前时间点
#include <windows.h>        // 包含Windows API,用于控制台操作
#include <conio.h>          // 包含控制台输入函数,用于检测按键using namespace std;        // 使用标准命名空间,简化代码int nScreenWidth = 120;     // 初始化屏幕宽度
int nScreenHeight = 30;     // 初始化屏幕高度struct snake {              // 定义贪吃蛇的身体段结构体int x;                  // 蛇段的横坐标int y;                  // 蛇段的纵坐标
};void GetConsoleSize(int& width, int& height) {  // 函数用于获取控制台窗口的尺寸CONSOLE_SCREEN_BUFFER_INFO csbi;           // 结构体变量,用于存储控制台屏幕缓冲区信息GetConsoleScreenBufferInfo(GetStdHandle(STD_OUTPUT_HANDLE), &csbi);  // 获取控制台屏幕缓冲区信息width = csbi.srWindow.Right - csbi.srWindow.Left + 1; // 计算窗口宽度height = csbi.srWindow.Bottom - csbi.srWindow.Top + 1; // 计算窗口高度
}int main() {                     // 主函数入口HANDLE hConsole = CreateConsoleScreenBuffer(GENERIC_READ | GENERIC_WRITE, 0, NULL, CONSOLE_TEXTMODE_BUFFER, NULL);  // 创建一个新的控制台屏幕缓冲区SetConsoleActiveScreenBuffer(hConsole);  // 设置当前控制台的活动屏幕缓冲区DWORD dwBytesWritten = 0;                 // 用于存储写入字符的数量while (true) {                           // 游戏主循环GetConsoleSize(nScreenWidth, nScreenHeight);  // 获取控制台的当前尺寸// Create display screenwchar_t* screen = new wchar_t[nScreenHeight * nScreenWidth];  // 动态分配一个二维数组用于显示屏幕for (int i = 0; i < nScreenHeight * nScreenWidth; i++) {      // 初始化屏幕数组为全空格screen[i] = L' ';}list<snake> snake1 = { {60,15},{61,15},{62,15},{63,15},{64,15},{65,15},{66,15},{67,15},{68,15},{69,15} };  // 初始化贪吃蛇的起始位置int nFoodX = 30;  // 食物的横坐标int nFoodY = 15;  // 食物的纵坐标int nScore = 0;   // 分数初始化为0int nSnakeDirection = 3;  // 贪吃蛇的方向初始化为向左bool bDead = false;       // 游戏结束标志初始化为未结束bool bKeyLeft = false, bKeyRight = false, bKeyLeftOld = false, bKeyRightOld = false;  // 键盘按键状态while (!bDead) {  // 当游戏未结束时循环auto t1 = chrono::system_clock::now();  // 获取当前时间点while ((chrono::system_clock::now() - t1) < ((nSnakeDirection % 2 == 1) ? 120ms : 200ms)) {  // 控制游戏帧率bKeyRight = (0x8000 & GetAsyncKeyState((unsigned char)('\x27'))) != 0;  // 检测右箭头键是否按下bKeyLeft = (0x8000 & GetAsyncKeyState((unsigned char)('\x25'))) != 0;   // 检测左箭头键是否按下if (bKeyRight && !bKeyRightOld) {  // 如果右键被按下且上次不是右键nSnakeDirection++;  // 改变方向if (nSnakeDirection == 4) nSnakeDirection = 0;  // 方向循环}if (bKeyLeft && !bKeyLeftOld) {  // 如果左键被按下且上次不是左键nSnakeDirection--;  // 改变方向if (nSnakeDirection == -1) nSnakeDirection = 3;  // 方向循环}bKeyRightOld = bKeyRight;  // 更新按键状态bKeyLeftOld = bKeyLeft;    // 更新按键状态}switch (nSnakeDirection) {  // 根据方向更新蛇头位置case 0: snake1.push_front({ snake1.front().x, snake1.front().y - 1 }); break;  // 向上case 1: snake1.push_front({ snake1.front().x + 1, snake1.front().y }); break;  // 向右case 2: snake1.push_front({ snake1.front().x, snake1.front().y + 1 }); break;  // 向下case 3: snake1.push_front({ snake1.front().x - 1, snake1.front().y }); break;  // 向左}// Check collision with foodif (snake1.front().x == nFoodX && snake1.front().y == nFoodY) {  // 如果蛇头碰到食物nScore++;  // 分数增加do {  // 重新生成食物的位置直到找到一个合适的空位nFoodX = rand() % nScreenWidth;nFoodY = (rand() % (nScreenHeight - 3)) + 3;} while (screen[nFoodY * nScreenWidth + nFoodX] != L' ');  // 检查新位置是否为空for (int i = 0; i < 5; i++)  // 增加蛇身长度snake1.push_back({ snake1.back().x, snake1.back().y });}// Check collision with wallsif (snake1.front().x < 0 || snake1.front().x >= nScreenWidth ||  // 检查蛇头是否撞墙snake1.front().y < 3 || snake1.front().y >= nScreenHeight) {bDead = true;  // 游戏结束}// Check collision with selffor (list<snake>::iterator i = next(snake1.begin()); i != snake1.end(); ++i)  // 检查蛇头是否碰到自己if (i->x == snake1.front().x && i->y == snake1.front().y)bDead = true;  // 游戏结束snake1.pop_back();  // 移除蛇尾// Clear screen and draw elementsfor (int i = 0; i < nScreenWidth * nScreenHeight; i++) screen[i] = L' ';  // 清屏for (int i = 0; i < nScreenWidth; i++) {  // 绘制边框screen[i] = L'=';  // 上方边框screen[2 * nScreenWidth + i] = L'=';  // 下方边框}wsprintf(&screen[nScreenWidth + 5], L"www.OneLoneCoder.com - S N A K E ! !                SCORE: %d", nScore);  // 显示分数for (const auto& s : snake1)  // 绘制蛇身screen[s.y * nScreenWidth + s.x] = bDead ? L'+' : L'O';screen[snake1.front().y * nScreenWidth + snake1.front().x] = bDead ? L'X' : L'@';  // 绘制蛇头screen[nFoodY * nScreenWidth + nFoodX] = L'%';  // 绘制食物if (bDead)  // 如果游戏结束wsprintf(&screen[15 * nScreenWidth + 40], L"    PRESS 'SPACE' TO PLAY AGAIN    ");  // 提示玩家按空格键重新开始WriteConsoleOutputCharacter(hConsole, screen, nScreenWidth * nScreenHeight, { 0,0 }, &dwBytesWritten);  // 刷新屏幕// Check for resizeint newWidth, newHeight;  // 新的宽高GetConsoleSize(newWidth, newHeight);  // 获取新的控制台尺寸if (newWidth != nScreenWidth || newHeight != nScreenHeight) {  // 如果尺寸改变bDead = true;  // 结束游戏}}delete[] screen;  // 释放屏幕数组内存// Wait for space to restartwhile ((0x8000 & GetAsyncKeyState((unsigned char)('\x20'))) == 0);  // 等待玩家按空格键重新开始}return 0;  // 返回程序执行状态
}

这篇关于GitHub经典贪吃蛇思路解析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

使用Python实现批量访问URL并解析XML响应功能

《使用Python实现批量访问URL并解析XML响应功能》在现代Web开发和数据抓取中,批量访问URL并解析响应内容是一个常见的需求,本文将详细介绍如何使用Python实现批量访问URL并解析XML响... 目录引言1. 背景与需求2. 工具方法实现2.1 单URL访问与解析代码实现代码说明2.2 示例调用

SSID究竟是什么? WiFi网络名称及工作方式解析

《SSID究竟是什么?WiFi网络名称及工作方式解析》SID可以看作是无线网络的名称,类似于有线网络中的网络名称或者路由器的名称,在无线网络中,设备通过SSID来识别和连接到特定的无线网络... 当提到 Wi-Fi 网络时,就避不开「SSID」这个术语。简单来说,SSID 就是 Wi-Fi 网络的名称。比如

SpringCloud配置动态更新原理解析

《SpringCloud配置动态更新原理解析》在微服务架构的浩瀚星海中,服务配置的动态更新如同魔法一般,能够让应用在不重启的情况下,实时响应配置的变更,SpringCloud作为微服务架构中的佼佼者,... 目录一、SpringBoot、Cloud配置的读取二、SpringCloud配置动态刷新三、更新@R

使用Java解析JSON数据并提取特定字段的实现步骤(以提取mailNo为例)

《使用Java解析JSON数据并提取特定字段的实现步骤(以提取mailNo为例)》在现代软件开发中,处理JSON数据是一项非常常见的任务,无论是从API接口获取数据,还是将数据存储为JSON格式,解析... 目录1. 背景介绍1.1 jsON简介1.2 实际案例2. 准备工作2.1 环境搭建2.1.1 添加

JAVA利用顺序表实现“杨辉三角”的思路及代码示例

《JAVA利用顺序表实现“杨辉三角”的思路及代码示例》杨辉三角形是中国古代数学的杰出研究成果之一,是我国北宋数学家贾宪于1050年首先发现并使用的,:本文主要介绍JAVA利用顺序表实现杨辉三角的思... 目录一:“杨辉三角”题目链接二:题解代码:三:题解思路:总结一:“杨辉三角”题目链接题目链接:点击这里

在C#中合并和解析相对路径方式

《在C#中合并和解析相对路径方式》Path类提供了几个用于操作文件路径的静态方法,其中包括Combine方法和GetFullPath方法,Combine方法将两个路径合并在一起,但不会解析包含相对元素... 目录C#合并和解析相对路径System.IO.Path类幸运的是总结C#合并和解析相对路径对于 C

Java解析JSON的六种方案

《Java解析JSON的六种方案》这篇文章介绍了6种JSON解析方案,包括Jackson、Gson、FastJSON、JsonPath、、手动解析,分别阐述了它们的功能特点、代码示例、高级功能、优缺点... 目录前言1. 使用 Jackson:业界标配功能特点代码示例高级功能优缺点2. 使用 Gson:轻量

Java如何接收并解析HL7协议数据

《Java如何接收并解析HL7协议数据》文章主要介绍了HL7协议及其在医疗行业中的应用,详细描述了如何配置环境、接收和解析数据,以及与前端进行交互的实现方法,文章还分享了使用7Edit工具进行调试的经... 目录一、前言二、正文1、环境配置2、数据接收:HL7Monitor3、数据解析:HL7Busines

python解析HTML并提取span标签中的文本

《python解析HTML并提取span标签中的文本》在网页开发和数据抓取过程中,我们经常需要从HTML页面中提取信息,尤其是span元素中的文本,span标签是一个行内元素,通常用于包装一小段文本或... 目录一、安装相关依赖二、html 页面结构三、使用 BeautifulSoup javascript

网页解析 lxml 库--实战

lxml库使用流程 lxml 是 Python 的第三方解析库,完全使用 Python 语言编写,它对 XPath表达式提供了良好的支 持,因此能够了高效地解析 HTML/XML 文档。本节讲解如何通过 lxml 库解析 HTML 文档。 pip install lxml lxm| 库提供了一个 etree 模块,该模块专门用来解析 HTML/XML 文档,下面来介绍一下 lxml 库