贪吃蛇游戏(代码篇)
我们并不是为了满足别人的期待而活着。
前言
这是我自己做的第五个小项目---贪吃蛇游戏(代码篇)。后期我会继续制作其他小项目并开源至博客上。
上一小项目是贪吃蛇游戏(必备知识篇),没看过的同学可以去看看:
有关贪吃蛇必备知识的小项目
https://blog.csdn.net/hsy1603914691/article/details/142455297?sharetype=blogdetail&sharerId=142455297&sharerefer=PC&sharesource=hsy1603914691&spm=1011.2480.3001.8118
实现代码
1. 下面代码直接复制即可运行。
2. 每个代码块都用简洁的总结和介绍。
<snake.h>文件
#define _CRT_SECURE_NO_WARNINGS
#include <locale.h>
#include <stdio.h>
#include <windows.h>
#include <stdbool.h>
#include <stdlib.h>
#include <time.h>#define KEY_PRESS(VK) ((GetAsyncKeyState(VK)&0x1) ? 1 : 0) //设置键值
#define POS_X 24 //蛇初始位置
#define POS_Y 5 //蛇初始位置//节点类型
typedef struct SnakeNode
{//节点的坐标int x;int y;//指向下一个节点的指针struct SnakeNode* next;
}SnakeNode;
typedef struct SnakeNode* pSnakeNode;//贪吃蛇的信息
typedef struct Snake
{pSnakeNode _pSnake;//贪吃蛇的身体节点pSnakeNode _pFood;//食物节点enum Direction _dir;//贪吃蛇的方向enum Game_Statues _status;//贪吃蛇的状态int _food_weight;//一个食物的分数int _score;//总分数int _sleep_time;//休息时间,即贪吃蛇的速度
}Snake;
typedef struct Snake* pSnake;//方向
enum Direction
{UP,DOWN,LEFT,RIGHT
};//状态
enum Game_Status
{OK,KILL_BY_WALL,KILL_BY_SELF,END_NORMAL
};//游戏开始
void GameStart(pSnake ps);
//欢迎函数
void WelcomeToGame();
//定位坐标
void SetPos(int x, int y);
//打印地图
void CreateMap();
//初始化贪吃蛇
void InitSnake(pSnake ps);
//创造食物
void CreateFood(pSnake ps);
//游戏运行
void GameRun(pSnake ps);
//打印帮助信息
void PrintHelpInfo();
//暂停设置
void Pause();
//实现贪吃蛇的移动
void SnakeMove(pSnake ps);
//判断是否吃到食物
int NextIsFood(pSnakeNode pn, pSnake ps);
//实现贪吃蛇吃食物并增长蛇身
void EatFood(pSnakeNode pn, pSnake ps);
//吃到食物后使食物消失
void NoFood(pSnakeNode pn, pSnake ps);
//被墙杀死
void KillByWall(pSnake ps);
//被自己杀死
void KillBySelf(pSnake ps);
//游戏结束
void GameEnd(pSnake ps);
<snake.c>文件
#include "snake.h"
//游戏开始
void GameStart(pSnake ps)
{//设置窗口system("mode con cols=100 lines=30");//调整CMD行与列system("title 贪吃蛇");//修改CMD的标题//获取标准输出的句柄,存放在houtput中。HANDLE houtput = GetStdHandle(STD_OUTPUT_HANDLE);if (houtput == INVALID_HANDLE_VALUE) // 处理错误,例如输出错误信息{fprintf(stderr, "Failed to get standard output handle.\n");return;}//创建一个CONSOLE_CURSOR_INFO的结构体CONSOLE_CURSOR_INFO CursorInfo;if (!GetConsoleCursorInfo(houtput, &CursorInfo)) // 处理错误,例如输出错误信息{fprintf(stderr, "Failed to get console cursor info.\n");return;}//隐藏控制台光标CursorInfo.bVisible = false; if (!SetConsoleCursorInfo(houtput, &CursorInfo)) // 处理错误,例如输出错误信息{fprintf(stderr, "Failed to set console cursor info.\n");return;}//欢迎函数WelcomeToGame();//打印地图CreateMap();//初始化贪吃蛇InitSnake(ps);//设置食物的位置CreateFood(ps);
}//欢迎函数
void WelcomeToGame()
{SetPos(32, 13);printf("Welcome to the Classic Snake Game!");SetPos(39, 22);system("pause");//打印完一个界面后直接暂停,直到点击继续system("cls");//在清空界面,打印新的一个界面SetPos(30, 13);wprintf(L"Navigate the Snake using ↑ ↓ ← →.");SetPos(33, 15);wprintf(L"Accelerate to earn more points.");SetPos(38, 23);system("pause");system("cls");
}//定位坐标
void SetPos(int x, int y)
{//获取标准输出的句柄,存放在houtput中HANDLE houtput = GetStdHandle(STD_OUTPUT_HANDLE);//设定我们想要定位的坐标COORD pos = { x,y };//将光标定位到pos2SetConsoleCursorPosition(houtput, pos);
}//打印地图
void CreateMap()
{int i = 0;//打印上边界for (i = 0; i < 29; i++){wprintf(L"□");}//打印下边界SetPos(0, 26);for (i=0; i < 29; i++){wprintf(L"□");}//打印左边界for (i = 1; i <= 25; i++){SetPos(0, i);wprintf(L"□");}//打印右边界for (i = 1; i <= 25; i++){SetPos(56, i);wprintf(L"□");}
}//初始化贪吃蛇
void InitSnake(pSnake ps)
{int i = 0;for (i = 0; i < 5; i++)//开始贪吃蛇一共设置五个长度{pSnakeNode cur = (pSnakeNode)malloc(sizeof(SnakeNode));if (cur == NULL){perror("InitSnake error");exit(1);}cur->next = NULL;cur->x = POS_X + 2 * i;cur->y = POS_Y;//头插法if (ps->_pSnake == NULL){ps->_pSnake = cur;}else{cur->next = ps->_pSnake;ps->_pSnake = cur;}}pSnakeNode cur = ps->_pSnake;while (cur != NULL){SetPos(cur->x, cur->y);wprintf(L"●");cur = cur->next;}//设置贪吃蛇的属性ps->_dir = RIGHT;ps->_score = 0;ps->_food_weight = 10;ps->_sleep_time = 200;ps->_status = OK;
}//设置食物的位置
void CreateFood(pSnake ps)
{int x;int y;
again:do{x = (rand()) % 53 + 2;y = (rand()) % 25 + 1;} while (x % 2 != 0);//不能与蛇身冲突pSnakeNode cur = ps->_pSnake;while (cur != NULL){if ((x == cur->x) && (y == cur->y)){goto again;}cur = cur->next;}//创建食物节点pSnakeNode pFood = (pSnakeNode)malloc(sizeof(SnakeNode));if (pFood == NULL){perror("CreateFood error");exit(1);}pFood->x = x;pFood->y = y;pFood->next = NULL;SetPos(x, y);wprintf(L"★");ps->_pFood = pFood;
}//游戏运行
void GameRun(pSnake ps)
{//打印欢迎界面PrintHelpInfo();//游戏开始运行do{//显示分数SetPos(64, 7);printf("Current score: %d", ps->_score);SetPos(64, 8);printf("Current food score: %2d", ps->_food_weight);//判断玩家操作if (KEY_PRESS(VK_UP) && (ps->_dir != UP)){ps->_dir = UP;}else if (KEY_PRESS(VK_DOWN) && (ps->_dir != DOWN)){ps->_dir = DOWN;}else if (KEY_PRESS(VK_LEFT) && (ps->_dir != LEFT)){ps->_dir = LEFT;}else if (KEY_PRESS(VK_RIGHT) && (ps->_dir != RIGHT)){ps->_dir = RIGHT;}else if (KEY_PRESS(VK_SPACE)){Pause();//暂停设置}else if (KEY_PRESS(VK_ESCAPE)){ps->_status = END_NORMAL;}else if (KEY_PRESS(VK_F3)){if (ps->_sleep_time > 80){ps->_sleep_time -= 30;ps->_food_weight += 2;}}else if (KEY_PRESS(VK_F4)){if (ps->_food_weight > 2){ps->_sleep_time += 30;ps->_food_weight -= 2;}}//实现贪吃蛇的移动SnakeMove(ps);//通过短暂暂停来展现动态效果Sleep(ps->_sleep_time);} while (ps->_status == OK);
}//打印欢迎界面
void PrintHelpInfo()
{SetPos(64, 10);wprintf(L"No wall passing. No self-biting.");SetPos(64, 12);wprintf(L"F3 to speed up. F4 to slow down.");SetPos(64, 14);wprintf(L"ESC to exit. Space to pause.");SetPos(74, 21);wprintf(L"Made by HSY,");SetPos(66, 22);wprintf(L"a uniquely independent pig.");
}//暂停设置
void Pause()
{while (1){Sleep(200);if (KEY_PRESS(VK_SPACE)){break;}}
}//实现贪吃蛇的移动
void SnakeMove(pSnake ps)
{pSnakeNode pNextNode = (pSnakeNode)malloc(sizeof(SnakeNode));if (pNextNode == NULL){perror("SnakeMove error");exit(1);}switch (ps->_dir){case UP:pNextNode->x = ps->_pSnake->x;pNextNode->y = ps->_pSnake->y - 1;break;case DOWN:pNextNode->x = ps->_pSnake->x;pNextNode->y = ps->_pSnake->y + 1;break;case LEFT:pNextNode->x = ps->_pSnake->x - 2;pNextNode->y = ps->_pSnake->y;break;case RIGHT:pNextNode->x = ps->_pSnake->x + 2;pNextNode->y = ps->_pSnake->y;break;}if (NextIsFood(pNextNode,ps)){EatFood(pNextNode, ps);}else{NoFood(pNextNode, ps);}KillByWall(ps);//被墙杀死KillBySelf(ps);//被自己杀死
}//判断是否吃到食物
int NextIsFood(pSnakeNode pn, pSnake ps)
{return (ps->_pFood->x == pn->x && ps->_pFood->y == pn->y);
}//实现贪吃蛇吃食物并增长蛇身
void EatFood(pSnakeNode pn, pSnake ps)
{ps->_pFood->next = ps->_pSnake;ps->_pSnake = ps->_pFood;free(pn);pn = NULL;pSnakeNode cur = ps->_pSnake;while (cur!=NULL){SetPos(cur->x, cur->y);wprintf(L"●");cur = cur->next;}ps->_score += ps->_food_weight;CreateFood(ps);
}//吃到食物后使食物消失
void NoFood(pSnakeNode pn, pSnake ps)
{pn->next = ps->_pSnake;ps->_pSnake = pn;pSnakeNode cur = ps->_pSnake;while (cur->next->next != NULL){SetPos(cur->x, cur->y);wprintf(L"●");cur = cur->next;}SetPos(cur->next->x, cur->next->y);printf(" ");free(cur->next);cur->next = NULL;
}//被墙杀死
void KillByWall(pSnake ps)
{if (ps->_pSnake->x == 0 || ps->_pSnake->x == 56 || ps->_pSnake->y == 0 || ps->_pSnake->y == 26){ps->_status = KILL_BY_WALL;}
}//被自己杀死
void KillBySelf(pSnake ps)
{pSnakeNode cur = ps->_pSnake->next;while (cur){if (cur->x == ps->_pSnake->x && cur->y == ps->_pSnake->y){ps->_status = KILL_BY_SELF;break;}cur = cur->next;}
}//游戏结束
void GameEnd(pSnake ps)
{//判断哪种结束方式switch (ps->_status){case END_NORMAL:SetPos(17, 12);printf("You have ended the game.");break;case KILL_BY_WALL:SetPos(10, 12);printf("You ended the game by hitting a wall.");break;case KILL_BY_SELF:SetPos(10, 12);printf("You ended the game by self-collision.");break;}//清除贪吃蛇pSnakeNode cur = ps->_pSnake;pSnakeNode prev = NULL;while (cur){prev = cur;cur = cur->next;free(prev);}}
<test.c>文件
#include "snake.h"
//游戏的主体进程
void test()
{char ch;do{system("cls");Snake snake = { 0 };GameStart(&snake);//游戏开始GameRun(&snake);//游戏运行GameEnd(&snake);//游戏结束SetPos(20, 15);//结束之后,询问是否再来一次printf("Play again? (Y/N)");ch = getchar();//用户输入一个字符并按回车后,实际上有两个字符进入了输入缓冲区:用户输入的字符和随后的换行符。第一个 getchar() 会读取用户输入的字符,而第二个 getchar() 则用来读取(并丢弃)换行符。getchar();} while (ch == 'Y'|| ch == 'y');SetPos(0, 28);//如果游戏结束,(为了美观)退出代码定位
}//主函数
int main()
{//设置本地环境setlocale(LC_ALL, "");//生成随机值srand((unsigned int)time(NULL));//测试游戏test();return 0;
}
致谢
感谢您花时间阅读这篇文章!如果您对本文有任何疑问、建议或是想要分享您的看法,请不要犹豫,在评论区留下您的宝贵意见。每一次互动都是我前进的动力,您的支持是我最大的鼓励。期待与您的交流,让我们共同成长,探索技术世界的无限可能!
相关文章:
贪吃蛇游戏(代码篇)
我们并不是为了满足别人的期待而活着。 前言 这是我自己做的第五个小项目---贪吃蛇游戏(代码篇)。后期我会继续制作其他小项目并开源至博客上。 上一小项目是贪吃蛇游戏(必备知识篇),没看过的同学可以去看看…...
数控走心机系统可以定制吗
当然,走心机系统是可以定制的。随着数控技术的不断发展,走心机的数控系统越来越灵活,可以根据用户的具体需求进行定制和优化。下面,我将从几个方面来详细解答这个问题: 一、系统定制的必要性 1. 满足不同加工需求…...
PHP实现OID(Object identifier)的编码和解码
转载于:https://bkssl.com/document/php_oid_encode_decode.html <?phpclass ASN1ObjectIdentifier {/*** OID字符串编码为二进制数据* param string $oid 字符串形式的OID* return string*/public static function encode($oid){$parts explode(., $oid);$pa…...
架构设计笔记-12-信息系统架构设计理论与实践
目录 知识要点 案例分析 1.Java企业级应用系统 2.c/s架构,b/s架构 知识要点 软件架构风格是描述某一特定应用领域中系统组织方式的惯用模式。架构风格定义了一类架构所共有的特征,主要包括架构定义、架构词汇表和架构约束。 数据挖掘是从数据库的大…...
【Power Compiler手册】15.多角多模式设计中的功耗优化
多角多模式设计中的功耗优化 可以使用多个运行条件和多种模式进行综合的设计被称为多角多模式设计。Design Compiler Graphical工具扩展了拓扑技术,以分析和优化这些设计。 有关多角多模式技术支持的综合工具的更多信息,请参见以下主题: • 优化多角多模式设计 • 报告命…...
关于HalconDeeplearn中的语义分割的实现
1.读取数据和数据集 read_dl_model (C:/Users/user/Desktop/大蒜测试/包裹/model_训练-240926-191345_opt.hdl, DLModelHandle) read_dict(C:/Users/user/Desktop/大蒜测试/包裹/model_训练-240926-162708_opt_dl_preprocess_params.hdict,[], [], DLDataset) 2.读取识别图片 I…...
【STL】AVLTree模拟实现
AVLTree模拟实现 1 前言2 AVL树的插入2.1 平衡因子不继续向上更新的情况2.2 平衡因子变为2或者-2,发生旋转2.2.1 左单旋2.2.2 右单旋2.2.3 左右双旋2.2.4 右左双旋 3 代码 1 前言 二叉搜索树的不足:如果出现极端情况,效率会变得很低。 AVL&am…...
无极低码课程【tomcat部署windows环境厂家乱码处理】
windows 下tomcat安装 下载地址一:https://tomcat.apache.org/download-90.cgi 下载地址二:https://archive.apache.org/dist/tomcat/ 解压tomcat,进入bin目录运行startup.bat...
注册安全分析报告:惠农网
前言 由于网站注册入口容易被黑客攻击,存在如下安全问题: 暴力破解密码,造成用户信息泄露短信盗刷的安全问题,影响业务及导致用户投诉带来经济损失,尤其是后付费客户,风险巨大,造成亏损无底洞…...
Qualitor checkAcesso.php 任意文件上传漏洞复现(CVE-2024-44849)
0x01 漏洞概述 Qualitor 8.24及之前版本存在任意文件上传漏洞,未经身份验证远程攻击者可利用该漏洞代码执行,写入WebShell,进一步控制服务器权限。 0x02 复现环境 FOFA:app="Qualitor-Web" 0x03 漏洞复现 PoC POST /html/ad/adfilestorage/request/checkAcess…...
PHP-FPM和FastCGI
文章目录 前言一. FastCGI1.定义2.工作方式3.协议4.架构5.工作原理(请求生命周期) 二. PHP-FPM1.定义:2.特性3.进程管理模式4.工作流程 三.关系与应用四.配置示例五.性能优化六.配置选项七.常见问题及解决方案 前言 PHP-FPM 是基于 FastCGI …...
【Linux快速入门(二)】Linux与ROS学习之编译基础(make编译)
目录 零.前置篇章 一.make的由来 二.安装make 三.编写Makefile 四.编译运行 五.删除可执行文件 零.前置篇章 第一篇【Linux快速入门】Linux与ROS学习之编译基础(gcc编译)_linuxros-CSDN博客 一.make的由来 "make"是一个用于自…...
jupyterlab的安装与使用攻略/包括汉化方法
官网链接 Project Jupyter | Home 1.第一步安装 打开控制台 使用pip工具安装 pip install jupyterlab 如图 2.安装成功后启动 jupyter lab 会自动启动它的web页面 然后就可以正常使用咯!! 如果需要更换浏览器访问 新开控制台执行下面命令 jupy…...
std::list
std::list是C标准库中的一个序列容器,它提供了双向链表的功能。std::list允许在序列的任何位置高效地插入和删除元素,而不会引起其他元素的移动,这使得std::list在需要频繁插入和删除操作的场景中非常有用。 std::list的特性: 双…...
opencv-rust 系列2: camera_calibration
opencv-rust 系列2: camera_calibration 前言: 这里只是opencv-rust自带示例的中文注解. 略微增加了一些代码也是我在调试时用到的. 说明: camera_calibration.rs是opencv-rust自带的示例, 在examples目录中可以找到,我增加了一些中文注释如下.如需运行可以在项目根目录执行命…...
JVM和GC案例详解
接上文JVM环境配置说明:上文博客 一、JVM远程连接设置 1. JMX方式连接(这种方式没有GC监控),设置如下 2. 连接成功后可以查看基础配置参数(和服务器配置一致) 2. jstatd方式连接(这种方式没有CPU监控) 添加jstatd方式连接 双击Tomcat࿰…...
postgreSql下载安装
一、下载 官网:PostgreSQL: The worlds most advanced open source database 二、安装 1.找到.exe文件,双击安装 2.跟着安装向导操作 三、启动...
GPT-SOVIT模型部署指南
一、模型介绍 强大的小样本语音转换和文本转语音 WebUI。 具有以下特征: 零样本 TTS: 输入 5 秒的声音样本并体验即时文本到语音的转换。少量样本 TTS: 仅使用 1 分钟的训练数据对模型进行微调,以提高语音相似度和真实感。跨语…...
怎么定时发朋友圈?
要实现微信朋友圈的定时发布,可以采用以下几种方法: 1、 绑定QQ号并使用QQ空间定时功能: 于微信和QQ的紧密联系,可以通过绑定QQ号,利用QQ空间的定时发布功能来间接实现微信朋友圈的定时发布。首先,在QQ空…...
如何利用phpstudy创建mysql数据库
phpStudy诞生于2007年,是一款老牌知名的PHP开发集成环境工具,产品历经多次迭代升级,目前有phpStudy经典版、phpStudy V8(2019版)等等,利用phpstudy可以快速搭建一个mysql环境,接下来我们就开始吧…...
生成xcframework
打包 XCFramework 的方法 XCFramework 是苹果推出的一种多平台二进制分发格式,可以包含多个架构和平台的代码。打包 XCFramework 通常用于分发库或框架。 使用 Xcode 命令行工具打包 通过 xcodebuild 命令可以打包 XCFramework。确保项目已经配置好需要支持的平台…...
阿里云ACP云计算备考笔记 (5)——弹性伸缩
目录 第一章 概述 第二章 弹性伸缩简介 1、弹性伸缩 2、垂直伸缩 3、优势 4、应用场景 ① 无规律的业务量波动 ② 有规律的业务量波动 ③ 无明显业务量波动 ④ 混合型业务 ⑤ 消息通知 ⑥ 生命周期挂钩 ⑦ 自定义方式 ⑧ 滚的升级 5、使用限制 第三章 主要定义 …...
AtCoder 第409场初级竞赛 A~E题解
A Conflict 【题目链接】 原题链接:A - Conflict 【考点】 枚举 【题目大意】 找到是否有两人都想要的物品。 【解析】 遍历两端字符串,只有在同时为 o 时输出 Yes 并结束程序,否则输出 No。 【难度】 GESP三级 【代码参考】 #i…...
关于iview组件中使用 table , 绑定序号分页后序号从1开始的解决方案
问题描述:iview使用table 中type: "index",分页之后 ,索引还是从1开始,试过绑定后台返回数据的id, 这种方法可行,就是后台返回数据的每个页面id都不完全是按照从1开始的升序,因此百度了下,找到了…...
DIY|Mac 搭建 ESP-IDF 开发环境及编译小智 AI
前一阵子在百度 AI 开发者大会上,看到基于小智 AI DIY 玩具的演示,感觉有点意思,想着自己也来试试。 如果只是想烧录现成的固件,乐鑫官方除了提供了 Windows 版本的 Flash 下载工具 之外,还提供了基于网页版的 ESP LA…...
Spring数据访问模块设计
前面我们已经完成了IoC和web模块的设计,聪明的码友立马就知道了,该到数据访问模块了,要不就这俩玩个6啊,查库势在必行,至此,它来了。 一、核心设计理念 1、痛点在哪 应用离不开数据(数据库、No…...
Linux 内存管理实战精讲:核心原理与面试常考点全解析
Linux 内存管理实战精讲:核心原理与面试常考点全解析 Linux 内核内存管理是系统设计中最复杂但也最核心的模块之一。它不仅支撑着虚拟内存机制、物理内存分配、进程隔离与资源复用,还直接决定系统运行的性能与稳定性。无论你是嵌入式开发者、内核调试工…...
C++课设:简易日历程序(支持传统节假日 + 二十四节气 + 个人纪念日管理)
名人说:路漫漫其修远兮,吾将上下而求索。—— 屈原《离骚》 创作者:Code_流苏(CSDN)(一个喜欢古诗词和编程的Coder😊) 专栏介绍:《编程项目实战》 目录 一、为什么要开发一个日历程序?1. 深入理解时间算法2. 练习面向对象设计3. 学习数据结构应用二、核心算法深度解析…...
第八部分:阶段项目 6:构建 React 前端应用
现在,是时候将你学到的 React 基础知识付诸实践,构建一个简单的前端应用来模拟与后端 API 的交互了。在这个阶段,你可以先使用模拟数据,或者如果你的后端 API(阶段项目 5)已经搭建好,可以直接连…...
6️⃣Go 语言中的哈希、加密与序列化:通往区块链世界的钥匙
Go 语言中的哈希、加密与序列化:通往区块链世界的钥匙 一、前言:离区块链还有多远? 区块链听起来可能遥不可及,似乎是只有密码学专家和资深工程师才能涉足的领域。但事实上,构建一个区块链的核心并不复杂,尤其当你已经掌握了一门系统编程语言,比如 Go。 要真正理解区…...
