基于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++中的虚拟继承的一些总结(虚拟继承,覆盖,派生,隐藏)

1.为什么要引入虚拟继承 虚拟继承是多重继承中特有的概念。虚拟基类是为解决多重继承而出现的。如:类D继承自类B1、B2,而类B1、B2都继承自类A,因此在类D中两次出现类A中的变量和函数。为了节省内存空间,可以将B1、B2对A的继承定义为虚拟继承,而A就成了虚拟基类。实现的代码如下: class A class B1:public virtual A; class B2:pu

C++对象布局及多态实现探索之内存布局(整理的很多链接)

本文通过观察对象的内存布局,跟踪函数调用的汇编代码。分析了C++对象内存的布局情况,虚函数的执行方式,以及虚继承,等等 文章链接:http://dev.yesky.com/254/2191254.shtml      论C/C++函数间动态内存的传递 (2005-07-30)   当你涉及到C/C++的核心编程的时候,你会无止境地与内存管理打交道。 文章链接:http://dev.yesky

C++的模板(八):子系统

平常所见的大部分模板代码,模板所传的参数类型,到了模板里面,或实例化为对象,或嵌入模板内部结构中,或在模板内又派生了子类。不管怎样,最终他们在模板内,直接或间接,都实例化成对象了。 但这不是唯一的用法。试想一下。如果在模板内限制调用参数类型的构造函数会发生什么?参数类的对象在模板内无法构造。他们只能从模板的成员函数传入。模板不保存这些对象或者只保存他们的指针。因为构造函数被分离,这些指针在模板外

C++工程编译链接错误汇总VisualStudio

目录 一些小的知识点 make工具 可以使用windows下的事件查看器崩溃的地方 dumpbin工具查看dll是32位还是64位的 _MSC_VER .cc 和.cpp 【VC++目录中的包含目录】 vs 【C/C++常规中的附加包含目录】——头文件所在目录如何怎么添加,添加了以后搜索头文件就会到这些个路径下搜索了 include<> 和 include"" WinMain 和

C/C++的编译和链接过程

目录 从源文件生成可执行文件(书中第2章) 1.Preprocessing预处理——预处理器cpp 2.Compilation编译——编译器cll ps:vs中优化选项设置 3.Assembly汇编——汇编器as ps:vs中汇编输出文件设置 4.Linking链接——链接器ld 符号 模块,库 链接过程——链接器 链接过程 1.简单链接的例子 2.链接过程 3.地址和

C++必修:模版的入门到实践

✨✨ 欢迎大家来到贝蒂大讲堂✨✨ 🎈🎈养成好习惯,先赞后看哦~🎈🎈 所属专栏:C++学习 贝蒂的主页:Betty’s blog 1. 泛型编程 首先让我们来思考一个问题,如何实现一个交换函数? void swap(int& x, int& y){int tmp = x;x = y;y = tmp;} 相信大家很快就能写出上面这段代码,但是如果要求这个交换函数支持字符型

零基础STM32单片机编程入门(一)初识STM32单片机

文章目录 一.概要二.单片机型号命名规则三.STM32F103系统架构四.STM32F103C8T6单片机启动流程五.STM32F103C8T6单片机主要外设资源六.编程过程中芯片数据手册的作用1.单片机外设资源情况2.STM32单片机内部框图3.STM32单片机管脚图4.STM32单片机每个管脚可配功能5.单片机功耗数据6.FALSH编程时间,擦写次数7.I/O高低电平电压表格8.外设接口

swiper实例

大家好,我是燐子,今天给大家带来swiper实例   微信小程序中的 swiper 组件是一种用于创建滑动视图的容器组件,常用于实现图片轮播、广告展示等效果。它通过一系列的子组件 swiper-item 来定义滑动视图的每一个页面。 基本用法   以下是一个简单的 swiper 示例代码:   WXML(页面结构) <swiper autoplay="true" interval="3

Java面试题:通过实例说明内连接、左外连接和右外连接的区别

在 SQL 中,连接(JOIN)用于在多个表之间组合行。最常用的连接类型是内连接(INNER JOIN)、左外连接(LEFT OUTER JOIN)和右外连接(RIGHT OUTER JOIN)。它们的主要区别在于它们如何处理表之间的匹配和不匹配行。下面是每种连接的详细说明和示例。 表示例 假设有两个表:Customers 和 Orders。 Customers CustomerIDCus

16.Spring前世今生与Spring编程思想

1.1.课程目标 1、通过对本章内容的学习,可以掌握Spring的基本架构及各子模块之间的依赖关系。 2、 了解Spring的发展历史,启发思维。 3、 对 Spring形成一个整体的认识,为之后的深入学习做铺垫。 4、 通过对本章内容的学习,可以了解Spring版本升级的规律,从而应用到自己的系统升级版本命名。 5、Spring编程思想总结。 1.2.内容定位 Spring使用经验