贪吃蛇游戏(代码篇)
我们并不是为了满足别人的期待而活着。
前言
这是我自己做的第五个小项目---贪吃蛇游戏(代码篇)。后期我会继续制作其他小项目并开源至博客上。
上一小项目是贪吃蛇游戏(必备知识篇),没看过的同学可以去看看:
有关贪吃蛇必备知识的小项目
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环境,接下来我们就开始吧…...

五、Linux之Vi和Vim编辑器
基本介绍 Vi Linux 系统会内置 vi 文本编辑 Vim 具有程序编辑的能力,可以看做是 Vi 的增强版本,可以主动的以字体颜色辨别语法的正确性,方便程序设计。 代码补完、编译及错误跳转等方便编程的功能特别丰富 常用的三种模式 正常模式 以 vim …...

git删除错误的commit
文章目录 1、git删除错误的commit2、.gitignore配置文件不生效的问题 1、git删除错误的commit git的流程如图: 当某次失误造成commit的版本有问题,需要回退到正常的版本修改后重新add。 首先通过git log查看commit提交记录,可以看到HEAD-…...

代码随想录算法训练营Day08 | 344.反转字符串、541. 反转字符串II、卡码网:54.替换数字
文章目录 344.反转字符串思路与重点 541. 反转字符串II思路与重点 卡码网:54.替换数字思路与重点 344.反转字符串 题目链接:344. 反转字符串 - 力扣(LeetCode)讲解链接:代码随想录 (programmercarl.com)状态ÿ…...

mysql锁之乐观锁、悲观锁、表锁、行锁、共享锁、排他锁
mysql锁之乐观锁、悲观锁、表锁、行锁、共享锁、排他锁 MySQL锁概述 锁是计算机协调多个进程或线程并发访问某一个资源的机制,在数据库中,除传统的计算资源(CPU、RAM、I/O)的争用以外,数据也是一种供许多用户共享的资…...

【软件干货】Android应用进程如何保活?
1.Android 应用进程保活方法介绍 在Android应用程序中,为了保证应用的正常运行和稳定性,有时需要对应用进程进行保活。以下是一些实现进程保活的方法: 1、使用前台服务(Foreground Service):将服务调用startForeground()方法&…...

neo4j部署保姆级教程
由于公司是基于大数据架构的,让部署neo4j数据库,之前没有接触过,然后紧急学了一下,并且从网上找了一些教程,决定还是记录下来,后续有时间了会在出一篇使用教程 环境准备(root用户) …...

【STM32CubeMX开发】-2.2-TIM_输出一个PWM信号
目录 1 Tim定时器的时钟源 2 Tim定时器的配置 2.1 PWM配置 2.2 中断配置 3 生成代码 4 测试结果 结尾 1 Tim定时器的时钟源 TIM3的时钟来源自APB1 Timer clocks,时钟树上所有总线频率均设置为了STM32F0能达到的最高频率,此时APB1 Timer clocks …...

Ngx+Lua+Redis 快速存储POST数据
系统几万台设备有windows有安卓还有linux系统,每个设备三分钟就会向服务器post设备的硬件信息,数据格式json,后台管理界面只需要最新的数据,不需要历史数据,业务逻辑非常简单,PHP代码就几行,已经…...

go-delve的使用
go-delve的非交互使用方式: dlv要执行的命令文件:cmd.dlv goroutines exit 执行非交互命令: yes n | dlv --allow-non-terminal-interactivetrue attach $pid --init cmd.dlv --end--...

Python网络爬虫技术详解
Python网络爬虫技术详解 引言 网络爬虫(Web Crawler),又称网络蜘蛛(Web Spider)或网络机器人(Web Robot),是一种按照一定规则自动抓取互联网信息的程序或脚本。它们通过遍历网页链…...