贪吃蛇游戏(代码篇)

我们并不是为了满足别人的期待而活着。

前言 

  这是我自己做的第五个小项目---贪吃蛇游戏(代码篇)。后期我会继续制作其他小项目并开源至博客上。 

  上一小项目是贪吃蛇游戏(必备知识篇),没看过的同学可以去看看:

有关贪吃蛇必备知识的小项目icon-default.png?t=O83Ahttps://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;
}

致谢

  感谢您花时间阅读这篇文章!如果您对本文有任何疑问、建议或是想要分享您的看法,请不要犹豫,在评论区留下您的宝贵意见。每一次互动都是我前进的动力,您的支持是我最大的鼓励。期待与您的交流,让我们共同成长,探索技术世界的无限可能!

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

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

相关文章

多态对象的存储方案小结

某个类型有几种不同的子类&#xff0c;Jackson中的JsonTypeInfo 和JsonSubTypes可以应对这种情形&#xff0c;但有点麻烦&#xff0c;并且name属性必须是字符串、必须用Jackson为基础的json工具类对json字符串和对象进行序列化和反序列化。用过一次这种方案后边就不想再用了。 …

【Python数据结构】深入理解Python中的列表、字典和集合!

【Python数据结构】深入理解Python中的列表、字典和集合&#xff01; 在Python编程中&#xff0c;数据结构是处理和组织数据的核心部分。Python提供了许多内置的数据结构&#xff0c;如列表、字典和集合&#xff0c;它们为程序员提供了灵活、高效的数据管理方式。本篇博客将深…

如何实现简单的 WinCC 项目分屏?

说明&#xff1a; 本文主要介绍了在不使用分屏器的情况下&#xff0c;通过 WinCC 项目中的设置&#xff0c;实现简单的分屏操作。两台显示器分别显示不同的 WinCC 画面&#xff0c;独自操作&#xff0c;互不影响。 试验环境 &#xff1a; 本文试验时所用硬件及软件环境…

Ubuntu如何显示pcl版本

终端输入&#xff1a; apt-cache show libpcl-dev可以看到&#xff0c;Ubuntu20.04&#xff0c;下载的pcl&#xff0c;应该都是1.10版本的

机器学习可解释性

机器学习的稳健性、可解释性和结果正确性等是人工智能安全可信应用必须解决的关键问题。 传统机器学习&#xff1a; 内置可解释性&#xff1a;决策树IF-Then规则&#xff0c;直观可理解事后可解释性&#xff1a;训练结束后的可解释技术特定于模型体系结构的解释与解释方法及模…

VS Code设置右侧滚动条高亮选中的字符位置

打开菜单“文件 -> 首选项 -> 设置”&#xff0c; 搜索“hightlight”,扩展中的Highlight Selections&#xff0c;再点击“在 settings.json中编辑”&#xff0c; 将editor.selectionHighlight选项改为true保存即可

【读书笔记·VLSI电路设计方法解密】问题12:制造MOSFET晶体管的主要工艺步骤是什么

VLSI芯片是在半导体材料上制造的,这种材料的导电性介于绝缘体和导体之间。通过一种称为掺杂的工艺引入杂质,可以改变半导体的电气特性。能够在半导体材料的细小且定义明确的区域内控制导电性,促使了半导体器件的发展。结合更简单的无源元件(电阻、电容和电感),这些器件被…

股市入门常见术语介绍

鉴于最近行情讨论火热&#xff0c;我也想借此平台&#xff0c;结合我大学时期身边同学老师的投资经历&#xff0c;写一篇交易入门术语简介。内容不多但是足以达到科普之用。 ​ 希望大家能谨慎对待投资&#xff0c;始终保持谦虚学习的态度。不要迷失在瞬息万变的金融市场&…

Redis拒绝连接问题分析与解决方案

目录 前言1. 问题描述2. Redis拒绝连接的常见原因分析2.1 Redis服务未启动2.2 Redis配置中的绑定地址问题2.3 防火墙或安全组问题2.4 Redis连接池耗尽2.5 Redis服务器负载过高2.6 权限配置问题 3. 深度解决方案和优化建议4. 总结 前言 在分布式系统中&#xff0c;Redis作为高性…

游戏服务端架构演进

文章目录 前言初出茅庐粗通皮毛略有小成炉火纯青内劲深厚最后 前言 对于网络游戏&#xff0c;一般分为客户端和服务端&#xff0c;客户端主要负责界面图像的渲染与一些交互操作&#xff0c;服务端主要负责数据的业务处理与存储还有与客户端之间的信息交互 比如玩家聊天、广播…

黑马程序员-redis项目实践笔记1

目录 一、 基于Session实现登录 发送验证码 验证用户输入验证码 校验登录状态 Redis代替Session登录 发送验证码修改 验证用户输入验证码 登录拦截器的优化 二、 商铺查询缓存 缓存更新策略 数据库和缓存不一致解决方案 缓存更新策略的最佳实践方案 实现商铺缓…

TS中如何正确处理window类型

在Typescript项目中&#xff0c;你可能都遇到过这个错误&#xff1a; Window & typeof globalThis 类型上不存在属性 X。 快速修复方案 我们将介绍几种不同的解决方案来解决这个问题。 Window 接口是在名为 lib.dom.d.ts 的文件中全局定义的。你可以使用各种技术来更改它&a…

Windows 11 24H2版本有哪些新功能_Windows 11 24H2十四大新功能介绍

距离上次发布的23H2版本已经过去了一年时间&#xff0c;现在&#xff0c;Win 11的24H2版本终于等到了&#xff0c;微软已经全面公开发布Win11 24H2版本&#xff0c;版本号为26100.1742&#xff0c;此次官宣的版本包括了消费者版、商业版、LTSC 2024版等&#xff0c;各种语言版本…

如何启动hive

检查mysql是否启动 通过Navicat测试mysql是否可以连接 找打hive配置文件所在目录 检查连接mysql的账号密码是否正确,如果不正确就要修改为正确的 初始化hive元数据存储的库:schematool -dbType <database_type> -initSchema 检查mysql中是否创建hive数据库,这里看到hive数…

zookeeper客户端

启动单机版的zookeeper 配置Maven环境 (1) IDEA自带maven (2) 更新Maven库镜像地址&#xff1a; ① 拷贝D:\Program Files\JetBrains\IntelliJ IDEA 2018.3.5\plugins\maven\lib\maven3\conf\settings.xml [IntelliJ的安装目录]到 C:/用户/username/.m2 (如果.m2文件不存在&…

华宇携司法大模型亮相2024中国移动全球合作伙伴大会

2024中国移动全球合作伙伴大会于10月11日在广州琶洲保利世贸博览馆盛大开幕。本届大会以“智焕新生 共创AI时代”为主题&#xff0c;深入探讨数据、算力与人工智能如何深度融合&#xff0c;全力推进AI规模发展、规模应用&#xff0c;加快形成AI技术能力、经济效益上的规模效应&…

第十三章 RabbitMQ之消息幂等性

目录 一、引言 二、消息幂等解决方案 2.1. 方案一 2.2. 方案二 一、引言 幂等是一个数学概念&#xff0c;用函数表达式来描述是这样的&#xff1a;f(x) f(f(x)) 。在程序开发中&#xff0c;则是指同一个业务&#xff0c;执行一次或多次对业务状态的影响是一致的。有些业务…

react实现实时计时的最简方式

js中时间的处理&#xff0c;不借助于moment/dayjs这样的工具库&#xff0c;原生获取格式化的时间&#xff0c;最简单的实现方式可以参考下面这样。 实现效果 代码实现 封装hooks import { useState, useEffect } from "react";export function useCountTime() {c…

Python酷库之旅-第三方库Pandas(150)

目录 一、用法精讲 681、pandas.Timestamp.now方法 681-1、语法 681-2、参数 681-3、功能 681-4、返回值 681-5、说明 681-6、用法 681-6-1、数据准备 681-6-2、代码示例 681-6-3、结果输出 682、pandas.Timestamp.replace方法 682-1、语法 682-2、参数 682-3、…

VUE 开发——Vue学习(三)—— 智慧商城项目

目录 解释各个模块 api接口模块&#xff1a;发送ajax请求的接口模块utils工具模块&#xff1a;自己封装的一些工具方法模块components组件模块&#xff1a;全局通用的组件router路由模块&#xff1a;封装要所有路由views&#xff1a;各个页面assets&#xff1a;各种资源 van…