基于easyx库的C/C++游戏编程实例-贪吃蛇|

2024-03-29 11:44

本文主要是介绍基于easyx库的C/C++游戏编程实例-贪吃蛇|,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

贪吃蛇|双人贪吃蛇游戏设计

1.设计物体结构

蛇结构:

struct SnakePlayer
{int size;int r;int dir;int speed;POINT pos[SNAKE_SIZE];bool live;int score;
}snake;

食物结构:

struct Food
{int x;int y;int r;bool flag;DWORD color;
}food,foodtest[FOOD_SIZE];
2.游戏初始化
void GameInit()
{//播放背景音乐mciSendString(_T("open ./res/Snake_BGM.mp3 alias BGM"), 0, 0, 0);mciSendString(_T("play BGM repeat"), 0, 0, 0);//创建一个窗口,控制窗口台是自动创建的,图形窗口是需要自己手动创建的(后加 SHOWCONSOLE)initgraph(WIN_WIDTH, WIN_HEIGHT);//加载开始界面loadimage(&bg, _T("./res/Snake_Background.jpg"));putimage(0, 0, &bg);//让随机数每次进入时都不同,只调用一次!!!srand(GetTickCount());//开始界面setbkmode(TRANSPARENT);//设置透明背景settextcolor(GREEN);settextstyle(60, 0, _T("黑体"));outtextxy(WIDTH / 2, HEIGHT / 2 - 70,_T("贪吃蛇" ));settextstyle(20, 0, _T("黑体"));outtextxy(WIDTH / 2 + 30, HEIGHT / 2 + 20, _T("按'C'开始游戏" ));while (!GetAsyncKeyState('C'));//蛇初始状态snake.size = SIZE_BEGIN;snake.r = R_BEGIN;snake.dir = RIGHT;snake.speed=SPEED_MIN;snake.live = true;snake.score = 0;for (int i = 0; i < snake.size; i++){snake.pos[i].x = X_BEGIN - 10 * i;snake.pos[i].y = Y_BEGIN;}//产生食物ProduceFood();
}
3.游戏界面加载
void GameDraw()
{BeginBatchDraw();setbkcolor(RGB(28, 115, 119));cleardevice();//是用当前背景色清空屏幕,并将当前点移至 (0, 0),EASYX自带//游戏外界面settextcolor(GREEN);settextstyle(50, 0, _T("黑体"));outtextxy(WIDTH + 10, 20, _T("贪吃蛇" ));settextstyle(20, 0, _T("黑体"));outtextxy(WIDTH + 10, 120, _T("开始/继续游戏——C"));outtextxy(WIDTH + 10, 150, _T("重新开始——M"));outtextxy(WIDTH + 10, 180, _T("暂停游戏——空格"));outtextxy(WIDTH + 10, 210,_T("上——W或↑"));outtextxy(WIDTH + 10, 240,_T("下——S或↓"));outtextxy(WIDTH + 10, 270, _T("左——A或←"));outtextxy(WIDTH + 10, 300,_T("右——D或→"));outtextxy(WIDTH + 10, 330, _T("加速——B"));//line(0, HEIGHT, WIDTH, HEIGHT);//画直线rectangle(0, 0, WIDTH, HEIGHT);//画矩形边框settextcolor(BLACK);settextstyle(20, 0, _T("黑体"));outtextxy(10, HEIGHT + 5,_T("当前分数:"));char t[SNAKE_SIZE];sprintf_s(t, "%d", snake.score);outtextxy(140, HEIGHT + 5, *t);for (int i = 0; i < snake.size; i++){setfillcolor(RED);solidcircle(snake.pos[i].x, snake.pos[i].y, snake.r);}if (food.flag){setfillcolor(food.color);solidcircle(food.x, food.y, food.r);}/*//测试用for (int j = 0; j < FOOD_SIZE; j++){if (foodtest[j].flag){setfillcolor(foodtest[j].color);solidcircle(foodtest[j].x, foodtest[j].y, foodtest[j].r);}}*/EndBatchDraw();
}
4.随机生成一个食物
void ProduceFood()
{food.r = rand() % 3 + 7;//随机生成食物位置,不能在蛇的身体上food.x = rand() % (WIDTH - 30) + food.r;food.y = rand() % (HEIGHT - 30) + food.r;while(1){int i = 0;for(; i < snake.size; i++){if (snake.pos[i].x + snake.r >= food.x - food.r && snake.pos[i].x - snake.r <= food.x + food.r&& snake.pos[i].y + snake.r >= food.y - food.r && snake.pos[i].y - snake.r <= food.y + food.r){food.x = rand() % (WIDTH - 30) + food.r;food.y = rand() % (HEIGHT - 30) + food.r;break;}}if (i == snake.size)break;}//随机生成食物颜色,不能与背景相同food.color = RGB(rand() % 256, rand() % 256, rand() % 256);while (food.color == RGB(28, 115, 119)){food.color = RGB(rand() % 256, rand() % 256, rand() % 256);}food.flag = true;
}
5.蛇吃食物机制
void EatFood()
{if (food.flag && snake.pos[0].x+snake.r > food.x - food.r && snake.pos[0].x - snake.r < food.x + food.r&& snake.pos[0].y + snake.r > food.y - food.r && snake.pos[0].y - snake.r < food.y + food.r){food.flag = false;snake.size++;snake.score++;}if (!food.flag){ProduceFood();}
}
6.控制蛇移动

蛇移动机制:

void SnakeMove()
{//蛇的身体移动for (int i = snake.size - 1; i > 0; i--){//Sleep(1);snake.pos[i] = snake.pos[i - 1];}//碰墙死亡机制switch (snake.dir){case UP:snake.pos[0].y -= snake.speed;break;case DOWN:snake.pos[0].y += snake.speed;break;case LEFT:snake.pos[0].x -= snake.speed;break;case RIGHT:snake.pos[0].x += snake.speed;break;}//穿墙所需机制/*switch (snake.dir){case UP:snake.pos[0].y -= speed;if (snake.pos[0].y < 0){snake.pos[0].y = HEIGHT;}break;case DOWN:snake.pos[0].y += speed;if (snake.pos[0].y > HEIGHT){snake.pos[0].y = 0;}break;case LEFT:snake.pos[0].x -= speed;if (snake.pos[0].x < 0){snake.pos[0].x = WIDTH;}break;case RIGHT:snake.pos[0].x += speed;if (snake.pos[0].x > WIDTH){snake.pos[0].x = 0;}break;}*/
}

按键控制:

void KeyCtrl()
{//Windows函数,	非阻塞//按键大写可检测到大小写,小写都检测不到//按键需要顺序+else if,这样目前检测连续按不能往回走if (GetAsyncKeyState(VK_UP) || GetAsyncKeyState('W')){if (snake.dir != DOWN){snake.dir = UP;}}else if (GetAsyncKeyState(VK_LEFT) || GetAsyncKeyState('A')){if (snake.dir != RIGHT){snake.dir = LEFT;}}else if (GetAsyncKeyState(VK_DOWN) || GetAsyncKeyState('S')){if (snake.dir != UP){snake.dir = DOWN;}}else if (GetAsyncKeyState(VK_RIGHT) || GetAsyncKeyState('D')){if (snake.dir != LEFT){snake.dir = RIGHT;}}if (GetAsyncKeyState('B')){snake.speed += SPEED_GAP;if (snake.speed>= SPEED_MAX){snake.speed = SPEED_MIN;}}if (GetAsyncKeyState(VK_SPACE)){while (!GetAsyncKeyState('C'));}
}
7.蛇碰撞死亡机制

bool Bump()
{//碰身体死亡for (int i = 4; i < snake.size; i++)	//最小情况为4{if (snake.pos[0].x == snake.pos[i].x && snake.pos[0].y == snake.pos[i].y){//snake.live = false;return false;}}//碰墙死亡if (snake.dir == LEFT && snake.pos[0].x - snake.r<0 || snake.dir == RIGHT && snake.pos[0].x + snake.r>WIDTH|| snake.dir == UP && snake.pos[0].y - snake.r<0 || snake.dir == DOWN && snake.pos[0].y + snake.r>HEIGHT){//snake.live = false;return false;}return true;	//	默认true,可不写
}
8.执行
int main()
{while (1){GameInit();while (1){//Producefoodtest();GameDraw();EatFood();KeyCtrl();SnakeMove();FlushBatchDraw();Sleep(100);	//可以让速度更慢if (!Bump() || GetAsyncKeyState('M'))break;}//播放死亡时的背景音乐mciSendString(_T("close BGM"), 0, 0, 0);mciSendString(_T("open ./res/Snake_Dead.mp3"), 0, 0, 0);mciSendString(_T("play ./res/Snake_Dead.mp3"), 0, 0, 0);settextcolor(RED);settextstyle(40, 0, _T("黑体"));outtextxy(WIDTH + 7, 400, _T("GAME OVER!"));settextstyle(20, 0, _T("黑体"));outtextxy(WIDTH + 10, 450, _T("获得总分:"));settextstyle(30, 0, _T("黑体"));char t[SNAKE_SIZE];sprintf_s(t, "%d", snake.score);outtextxy(WIDTH + 140, 445, *t);while (!GetAsyncKeyState('M'));//停住播放。不能和play相邻,它会立即关闭,因为程序执行太快!mciSendString(_T("close ./res/Snake_Dead.mp3"), 0, 0, 0);}return 0;
}

结果演示视频:

可见,我还对游戏画面进行了优化,添加了文字描述,画面较为美观。

基于此设计,我们可以衍生出双人贪吃蛇的游戏设计,这只不过是在贪吃蛇设计上添加一个蛇,再修改一下碰撞机制,比如说双方碰撞检测:

for (int k = 1; k < snake[i].size; k++){if (i + 1 < SNAKE_NUM&&snake[i].live && snake[i+1].live){if (snake[i].pos[0].x - snake[i].r < snake[i + 1].pos[k].x + snake[i].r && snake[i].pos[0].x + snake[i].r > snake[i + 1].pos[k].x - snake[i].r&& snake[i].pos[0].y - snake[i].r < snake[i + 1].pos[k].y + snake[i].r && snake[i].pos[0].y + snake[i].r > snake[i + 1].pos[k].y - snake[i].r){snake[i].live = false;DeadFood(i);}}if ((i - 1) >= 0&&snake[i].live && snake[i - 1].live){if (snake[i].pos[0].x - snake[i].r < snake[i - 1].pos[k].x + snake[i].r && snake[i].pos[0].x + snake[i].r > snake[i - 1].pos[k].x - snake[i].r&& snake[i].pos[0].y - snake[i].r < snake[i - 1].pos[k].y + snake[i].r && snake[i].pos[0].y + snake[i].r > snake[i - 1].pos[k].y - snake[i].r){snake[i].live = false;DeadFood(i);}}}

还可以添加一些玩法,增加游戏的可玩性。此外,可以设计游戏初始画面和结束画面,添加鼠标控制等。以下是我的双人贪吃蛇设计:

这篇关于基于easyx库的C/C++游戏编程实例-贪吃蛇|的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

【C++ Primer Plus习题】13.4

大家好,这里是国中之林! ❥前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站。有兴趣的可以点点进去看看← 问题: 解答: main.cpp #include <iostream>#include "port.h"int main() {Port p1;Port p2("Abc", "Bcc", 30);std::cout <<

C++包装器

包装器 在 C++ 中,“包装器”通常指的是一种设计模式或编程技巧,用于封装其他代码或对象,使其更易于使用、管理或扩展。包装器的概念在编程中非常普遍,可以用于函数、类、库等多个方面。下面是几个常见的 “包装器” 类型: 1. 函数包装器 函数包装器用于封装一个或多个函数,使其接口更统一或更便于调用。例如,std::function 是一个通用的函数包装器,它可以存储任意可调用对象(函数、函数

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

Linux 网络编程 --- 应用层

一、自定义协议和序列化反序列化 代码: 序列化反序列化实现网络版本计算器 二、HTTP协议 1、谈两个简单的预备知识 https://www.baidu.com/ --- 域名 --- 域名解析 --- IP地址 http的端口号为80端口,https的端口号为443 url为统一资源定位符。CSDNhttps://mp.csdn.net/mp_blog/creation/editor

【Python编程】Linux创建虚拟环境并配置与notebook相连接

1.创建 使用 venv 创建虚拟环境。例如,在当前目录下创建一个名为 myenv 的虚拟环境: python3 -m venv myenv 2.激活 激活虚拟环境使其成为当前终端会话的活动环境。运行: source myenv/bin/activate 3.与notebook连接 在虚拟环境中,使用 pip 安装 Jupyter 和 ipykernel: pip instal

06 C++Lambda表达式

lambda表达式的定义 没有显式模版形参的lambda表达式 [捕获] 前属性 (形参列表) 说明符 异常 后属性 尾随类型 约束 {函数体} 有显式模版形参的lambda表达式 [捕获] <模版形参> 模版约束 前属性 (形参列表) 说明符 异常 后属性 尾随类型 约束 {函数体} 含义 捕获:包含零个或者多个捕获符的逗号分隔列表 模板形参:用于泛型lambda提供个模板形参的名

【机器学习】高斯过程的基本概念和应用领域以及在python中的实例

引言 高斯过程(Gaussian Process,简称GP)是一种概率模型,用于描述一组随机变量的联合概率分布,其中任何一个有限维度的子集都具有高斯分布 文章目录 引言一、高斯过程1.1 基本定义1.1.1 随机过程1.1.2 高斯分布 1.2 高斯过程的特性1.2.1 联合高斯性1.2.2 均值函数1.2.3 协方差函数(或核函数) 1.3 核函数1.4 高斯过程回归(Gauss

6.1.数据结构-c/c++堆详解下篇(堆排序,TopK问题)

上篇:6.1.数据结构-c/c++模拟实现堆上篇(向下,上调整算法,建堆,增删数据)-CSDN博客 本章重点 1.使用堆来完成堆排序 2.使用堆解决TopK问题 目录 一.堆排序 1.1 思路 1.2 代码 1.3 简单测试 二.TopK问题 2.1 思路(求最小): 2.2 C语言代码(手写堆) 2.3 C++代码(使用优先级队列 priority_queue)

【C++高阶】C++类型转换全攻略:深入理解并高效应用

📝个人主页🌹:Eternity._ ⏩收录专栏⏪:C++ “ 登神长阶 ” 🤡往期回顾🤡:C++ 智能指针 🌹🌹期待您的关注 🌹🌹 ❀C++的类型转换 📒1. C语言中的类型转换📚2. C++强制类型转换⛰️static_cast🌞reinterpret_cast⭐const_cast🍁dynamic_cast 📜3. C++强制类型转换的原因📝