贪吃蛇完整代码

贪吃蛇代码

Snake.h

//Snake.h
#include<windows.h>
#include<stdbool.h>
#include<stdlib.h>
#include<stdio.h>
#include<time.h>#define POS_X 24
#define POS_Y 5#define WALL L'□'
#define BODY L'●'
#define FOOD L'★'
//类型的声明
// 
//蛇身的结点类型
typedef struct Snakenode
{//坐标int x;int y;//指向下一个节点的指针struct Snakenode* next;
}Snakenode,*pSnakeNode;//结构体指针pSnakeNode//贪吃蛇
typedef struct Snake
{pSnakeNode _pSnake;//指向蛇头的指针pSnakeNode _pFood;//指向食物节点的指针enum DIRECTION _dir;//蛇的方向enum GAME_STATUS _status;//游戏的状态int _food_weight;//食物分数int _score;//总分数int _sleep_time;//休息时间  时间越短 速度越快  时间越长 速度越慢
}Snake,*pSnake;//枚举方向
enum DIRECTION
{UP = 1,DOWN,LEFT,RIGHT
};//蛇的状态
//正常 撞墙 撞到自己 正常退出
enum GAME_STATUS 
{OK,KILL_BY_WALL,KILL_BY_SELF,END_NORMAL
};//函数的声明//定位光标位置
void SetPos(short x, short y);//游戏的初始化
void GameStart(pSnake ps);//欢迎界面的打印
void WelcomeToGame();//创建地图
void CreatMap();//4.创建蛇 --初始化蛇身
void InitSnake(pSnake ps);//创建食物
void CreateFood(pSnake ps);//游戏运行逻辑
void GameRun(pSnake ps);//蛇走一步的过程
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

//snake.c
#include"snake.h"void SetPos(short x, short y)//封装一个函数SetPos 光标位置
{HANDLE houtput = NULL;//获取标准输出的句柄(⽤来标识不同设备的数值)houtput = GetStdHandle(STD_OUTPUT_HANDLE);//定位光标位置COORD pos = { x, y };SetConsoleCursorPosition(houtput, pos);
}
void WelcomeToGame()
{SetPos(45, 14);wprintf(L"欢迎来到贪吃蛇小游戏\n");SetPos(46, 20);system("pause");system("cls");//清除控制台信息SetPos(30, 14);wprintf(L"用↑   ↓   ←   → 来控制蛇的移动 按F3加速 F4减速\n");SetPos(34, 15);wprintf(L"加速能够得到更高的分数");SetPos(46, 20);system("pause");system("cls");//紧接着下面应该是地图绘制
}void CreatMap()
{//27行 58列的棋盘//上int i = 0;for (i = 0; i < 29; i++){//□wprintf(L"%lc", WALL);}//下SetPos(0, 26);for (i = 0; i < 29; i++)wprintf(L"%lc", WALL);//左for (i = 1; i <= 25; i++){SetPos(0, i);wprintf(L"%lc", WALL);}//右for (i = 1; i <= 25; i++){SetPos(56, i);wprintf(L"%lc", WALL);}//getchar();
}//4.创建蛇 --初始化蛇身
void InitSnake(pSnake ps)
{int i = 0;pSnakeNode cur = NULL;for (i = 0; i < 5; i++){cur = (pSnakeNode)malloc(sizeof(Snakenode));if (cur == NULL){perror("InitSnake()::malloc()");return;}cur->next = NULL;cur->x =POS_X +i*2;//显示 + 未使用的表达式结果cur->y = POS_Y;//头插法插入链表if (ps->_pSnake == NULL)//空链表  //蛇头{ps->_pSnake = cur;}else//非空{cur->next = ps->_pSnake;ps->_pSnake = cur;}}cur = ps->_pSnake;while (cur){SetPos(cur->x, cur->y);wprintf(L"%lc", BODY);cur = cur->next;}//设置贪吃蛇的属性ps->_dir = RIGHT;//默认向右ps->_score = 0;ps->_food_weight = 10;ps->_sleep_time = 200;//毫秒ps->_status = OK;/*getchar();*/为什么我的代码只打印一个●
}void CreateFood(pSnake ps)
{int x = 0;int y = 0;//x = rand()%53+2;//x:2~54---0~52 //还要保证x是偶数//y = rand()%25+1;//y:1~25---0~24
again:do{x = rand() % 53 + 2;//x:2~54---0~52 //还要保证x是偶数y = rand()%25+1;//y:1~25---0~24} while (x % 2 != 0);//x和y的坐标不能跟蛇的身体相冲突pSnakeNode cur = ps->_pSnake;while (cur){if (x == cur->x && y == cur->y){goto again;}cur = cur->next;}//创捷食物结点pSnakeNode pFood=(pSnakeNode)malloc(sizeof(Snakenode));if (pFood == NULL){perror("CreateFood()::malloc()");}pFood->x = x;pFood->y = y;pFood->next = NULL;SetPos(x, y);wprintf(L"%lc", FOOD);ps->_pFood = pFood;
}void GameStart(pSnake ps)
{//0.先设置窗口的大小 然后把光标隐藏system("mode con cols=100 lines=30");//30行 100列system("title 贪吃蛇");HANDLE houtput=GetStdHandle(STD_OUTPUT_HANDLE);//隐藏光标操作//定义一个光标信息的结构体CONSOLE_CURSOR_INFO cursor_info;//获取和houtput句柄相关的控制台上的光标信息 并存放在cursor_info中GetConsoleCursorInfo(houtput, &cursor_info);//获取控制台光标信息cursor_info.bVisible = false;//设置和houtput句柄相关的控制台上的光标信息SetConsoleCursorInfo(houtput, &cursor_info);//1.打印欢迎页面  //2.功能介绍WelcomeToGame();//3.绘制地图CreatMap();//4.创建蛇InitSnake(ps);//5.创建食物CreateFood(ps);
}void PrintHelpInfo()
{SetPos(64, 10);wprintf(L"%ls", L"不能穿墙,不能咬到自己");SetPos(64, 11);wprintf(L"%ls", L"用↑   ↓   ←   → 来控制蛇的移动 \n");SetPos(64, 12);wprintf(L"%ls", L"按F3加速 F4减速");SetPos(64, 13);wprintf(L"%ls", L"按ESC退出游戏  空格暂停游戏");SetPos(64, 14);wprintf(L"%ls", L"made in LiFeiFei");
}//获取按键情况 如果按键最小位为1 说明已按过
#define KEY_PRESS(VK) ( (GetAsyncKeyState(VK) & 0x1) ? 1 : 0 )void pause()//暂停
{while (1){Sleep(200);if (KEY_PRESS(VK_SPACE)){break;}}
}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){SetPos(cur->x, cur->y);wprintf(L"%lc", BODY);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"%lc", BODY);cur = cur->next;}//把最后一个结点打印成空格SetPos(cur->next->x, cur->next->y);printf("  ");//释放最后一个结点free(cur->next);//将倒数第二个结点的地址域置为NULLcur->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;//cur指向蛇身while (cur){if (cur->x == ps->_pSnake->x && cur->y == ps->_pSnake->y){ps->_status = KILL_BY_SELF;break;}cur = cur->next;}
}void SnakeMove(pSnake ps)
{//创建一个结点 表示蛇即将到的下一个结点pSnakeNode pNextNode=(pSnakeNode)malloc(sizeof(Snakenode));if (pNextNode == NULL){perror("SnakeMove()::malloc()");return;}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);
}void GameRun(pSnake ps)
{//打印帮助信息PrintHelpInfo();do{//打印总分数 和食物的分值SetPos(64, 8);printf("总分数:%d\n", ps->_score);SetPos(64, 9);printf("当前食物的分值:%2d\n", ps->_food_weight);if (KEY_PRESS(VK_UP) && ps->_dir != DOWN){ps->_dir = UP;}else if (KEY_PRESS(VK_DOWN) && ps->_dir != UP){ps->_dir = DOWN;}else if (KEY_PRESS(VK_LEFT) && ps->_dir != RIGHT){ps->_dir = LEFT;}else if (KEY_PRESS(VK_RIGHT) && ps->_dir != LEFT){ps->_dir = RIGHT;}else if (KEY_PRESS(VK_SPACE))//空格{pause();}else if (KEY_PRESS(VK_ESCAPE))//ESC{//正常退出游戏ps->_status = END_NORMAL;//退出游戏break;}else if (KEY_PRESS(VK_F3))//F3加速---休眠时间减少{if (ps->_sleep_time >= 80)//初始休眠时间为200-80=120 减4次{//最小休眠时间为50ps->_sleep_time -= 30;ps->_food_weight += 2;//食物初始分值为10  ⼀个⻝物分数最⾼是20分}}else if (KEY_PRESS(VK_F4))//F4减速{if (ps->_sleep_time < 320)//200 230 260 290 320{						  //10   8   6   4   2ps->_sleep_time += 30;ps->_food_weight -= 2;//⼀个⻝物分数最低是2分}}SnakeMove(ps);//蛇走一步的过程Sleep(ps->_sleep_time);} while (ps->_status == OK);
}void GameEnd(pSnake ps)
{SetPos(24, 12);switch (ps->_status){case END_NORMAL:printf("有事 不玩了\n");break;case KILL_BY_WALL:printf("撞个墙玩玩\n");break;case KILL_BY_SELF:printf("吃个自己吧\n");break;}//释放蛇身的链表pSnakeNode cur = ps->_pSnake;while (cur){pSnakeNode del = cur;cur = cur->next;free(del);}
}

test.c

//test.c
#include<locale.h>
#include"snake.h"
//完成的是游戏的测试逻辑
void test()
{int ch = 0;do{//创建贪吃蛇Snake snake = { 0 };//初始化游戏 //1.打印环境页面//2.功能介绍//3.绘制地图//4.创建蛇//5.创建食物//6.设置游戏的相关信息GameStart(&snake);//运行游戏GameRun(&snake);//结束游戏--善后工作GameEnd(&snake);SetPos(20, 15);printf("再来一局吗?(Y/N):");ch = getchar();//按下回车后 其实接收到的是\n 所以需要清理掉\n//getchar();//清理掉\nwhile (getchar() != '\n');} while (ch=='Y' || ch == 'y');SetPos(0, 26);//让代码结束的信息放在地图外 
}
int main()
{//设置适配本地环境setlocale(LC_ALL, "");//中文 宽字符的打印srand((unsigned int)time(NULL));test();return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/2281.shtml

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

相关文章

linux复习提纲

后台运行 &&#xff1a;命令行尾输入&#xff0c;命令执行后台运行fg % 作业号&#xff1a;作业前台执行bg % 作业号&#xff1a;作业后台执行 用户权限 主user&#xff08;u&#xff09;-同一组的&#xff08;g&#xff09;-其他&#xff08;o&#xff09; rwx&#xf…

【YOLOv8改进[检测头Head]】YOLOv8的“新头”之动态头(DynamicHead)

目录 一 DynamicHead 二 YOLOv8的“新头”之动态头 1 总体修改 2 配置文件 3 训练 其他 一 DynamicHead 官方论文地址&#xff1a;https://arxiv.org/pdf/2106.08322.pdf 官方代码地址&#xff1a;GitCode - 开发者的代码家园 在计算机视觉应用中&#xff0c;目标检测…

软件著作权登记材料

1、计算机软件著作权登记申请表 2、申请者身份证明 (1)著作权人为企业法人的&#xff1a; 提交有效的企业法人营业执照或事业单位法人证书副本复印件&#xff0c;并需加盖单位公章; (2)著作权人为自然人的&#xff1a; ?提交有效的自然人身份证复印件(正反面复印)。提交《非职…

第四百七十五回

文章目录 1. 概念介绍2. 功能与用法2.1 主要功能2.2 使用方法 3. 示例代码4. 内容总结 我们在上一章回中介绍了"四个bublue包对比与总结"相关的内容&#xff0c;本章回中将介绍Get包.闲话休提&#xff0c;让我们一起Talk Flutter吧。 1. 概念介绍 我们在本章回中将介…

zustand关于状态变化,是写在内部,还是外部

在使用Zustand时&#xff0c;通常建议将逻辑写在Zustand内部。Zustand是一个状态管理库&#xff0c;它提供了一种简单且强大的方式来管理状态&#xff0c;并且它的设计初衷就是为了让状态管理更加简洁和直观。 在Zustand内部定义状态和操作函数&#xff0c;并通过useStore hoo…

文件、文件夹被占用无法删除

一、针对文件夹无法删除 1、我们打开这个无法删除的文件夹&#xff0c;复制这个文件夹的地址。 2、打开windows任务管理器&#xff0c;选择“性能”选项卡&#xff0c;点击“打开资源监视器”。 3、在“资源监视器”中&#xff0c;cpu中找到“关联的句柄”&#xff0c;在文本框…

4.12 SpringBoot整合AOP ❤❤❤

1. AOP简介 要介绍面向切面编程(Aspect-Oriented Programming,AOP),需要读者 首先考虑这样一个场景&#xff1a; 公司有一个人力资源管理系统目前已经上线&#xff0c;但是系统运行不稳定&#xff0c;有时运行得很慢&#xff0c;为了检测出到底是哪个环节出问题了&#xff0c;…

错了,全部错了

感觉还是挺有难度的吧。。 其实最小割在这种最大/最小化代价的问题上的拓展性是比最大流要好的&#xff0c;甚至有些问题用最大流建模后不好求而要转化为求最小割&#xff08;例如 CF1368H Breadboard Capacity&#xff0c;但是也不绝对&#xff0c;有极少数题是用最小割建模…

力扣HOT100 - 2. 两数相加

解题思路&#xff1a; 缺位的节点进行补零处理&#xff0c;如97323补充为973023 注意相加的进位问题 class Solution {public ListNode addTwoNumbers(ListNode l1, ListNode l2) {ListNode head null, tail null;int carry 0;while (l1 ! null || l2 ! null) {int n1 l…

2013-2021年各省经济韧性相关测度指标面板数据

2013-2021年各省经济韧性相关测度指标面板数据 1、时间&#xff1a;2013-2021年 2、指标&#xff1a;城镇化率 %、财政科学技术支出&#xff08;亿元&#xff09;、万人高等教育在校人数&#xff08;万人&#xff09;、财政教育支出&#xff08;亿元&#xff09;、第三产业占…

semaphore信号量使用+原理分析

1.概述 Semaphore 信号量&#xff0c;相当于一个计数器&#xff0c;通常用来限制线程的数量。 每个线程操作前会先获取一个许可证&#xff0c;逻辑处理完成之后就归还这个许可证。 通俗的解释&#xff1a;相当于一个停车场&#xff0c;有10个停车位&#xff0c;进来一个车&am…

QtCharts 组件

Qtcharts 组件基于GraphicsView模式实现&#xff0c;其核心是QChartView和QChart的二次封装版。 在pro文件中包含QT charts来引入绘图类库。 头文件中定义QT_CHARTS_USE_NAMESPACE宏&#xff0c;这样才可以正常的使用绘图功能。 一般情况下我们会在mainwindows.h头文件中增…

本地部署Docker容器可视化图形管理工具DockerUI并实现无公网IP远程访问——“cpolar内网穿透”

文章目录 前言1. 安装部署DockerUI2. 安装cpolar内网穿透3. 配置DockerUI公网访问地址4. 公网远程访问DockerUI5. 固定DockerUI公网地址 前言 DockerUI是一个docker容器镜像的可视化图形化管理工具。DockerUI可以用来轻松构建、管理和维护docker环境。它是完全开源且免费的。基…

stack、queue(priority_queue)的模拟实现和deque的简单介绍

stack和queue(priority_queue) 1. 容器适配器 适配器(Adapter)&#xff1a;一种用来修饰容器(Containers)或仿函数(Functors)或迭代器(Iterator)接口的东西。 适配器是一种设计模式&#xff0c;该模式将一个类的接口转换成客户希望的另外一个接口。 现实中拿插座来说&#xf…

LeetCode 每日一题 Day 137-143

928. 尽量减少恶意软件的传播 II(Hard) 给定一个由 n 个节点组成的网络&#xff0c;用 n x n 个邻接矩阵 graph 表示。在节点网络中&#xff0c;只有当 graph[i][j] 1 时&#xff0c;节点 i 能够直接连接到另一个节点 j。 一些节点 initial 最初被恶意软件感染。只要两个节点…

llm构建知识库

构建词向量库 在RAG应用中&#xff0c;我们需要大模型调用知识库中的专业内容对用户的提问进行更加完善的回答&#xff0c;因此如何为llm引入知识库便尤为关键&#xff0c;在本篇内容中&#xff0c;我会从基础的词向量开始介绍&#xff0c;最终到如何完成一个可供llm调用的向量…

C语言进阶课程学习记录- 函数与宏分析

C语言进阶课程学习记录- 函数与宏分析 实验-宏和函数实验-宏的副作用实验-宏的妙用小结 本文学习自狄泰软件学院 唐佐林老师的 C语言进阶课程&#xff0c;图片全部来源于课程PPT&#xff0c;仅用于个人学习记录 实验-宏和函数 #include <stdio.h>#define RESET(p, len) …

程序员开发必备,开发资源资料分享【3】

第2部分内容 090-100036401-专栏课-聂鹏程-分布式技术原理与算法解析&#xff08;完结&#xff09;提取码&#xff1a; 091-100036501-专栏课-王健-说透中台&#xff08;完结&#xff09;提取码&#xff1a; 092-100036601-专栏课-石雪峰-DevOps 实战笔记&#xff08;完结&a…

【Nginx】centos和Ubuntu操作系统下载Nginx配置文件并启动Nginx服务详解

目录 &#x1f337; 安装Nginx环境 &#x1f340; centos操作系统 &#x1f340; ubuntu操作系统 &#x1f337; 安装Nginx环境 以下是在linux系统中安装Nginx的步骤&#xff1a; 查看服务器属于哪个操作系统 cat /etc/os-release安装 yum&#xff1a; 如果你确定你的系统…

Spring是如何解决循环依赖的

简介 所谓循环依赖指的是&#xff1a;BeanA对象的创建依赖于BeanB&#xff0c;BeanB对象的创建也依赖于BeanA&#xff0c;这就造成了死循环&#xff0c;如果不做处理的话势必会造成栈溢出。Spring通过提前曝光机制&#xff0c;利用三级缓存解决循环依赖问题。 变量描述single…