基于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++统计函数执行时间的最佳实践

《C++统计函数执行时间的最佳实践》在软件开发过程中,性能分析是优化程序的重要环节,了解函数的执行时间分布对于识别性能瓶颈至关重要,本文将分享一个C++函数执行时间统计工具,希望对大家有所帮助... 目录前言工具特性核心设计1. 数据结构设计2. 单例模式管理器3. RAII自动计时使用方法基本用法高级用法

深入解析C++ 中std::map内存管理

《深入解析C++中std::map内存管理》文章详解C++std::map内存管理,指出clear()仅删除元素可能不释放底层内存,建议用swap()与空map交换以彻底释放,针对指针类型需手动de... 目录1️、基本清空std::map2️、使用 swap 彻底释放内存3️、map 中存储指针类型的对象

Python异步编程之await与asyncio基本用法详解

《Python异步编程之await与asyncio基本用法详解》在Python中,await和asyncio是异步编程的核心工具,用于高效处理I/O密集型任务(如网络请求、文件读写、数据库操作等),接... 目录一、核心概念二、使用场景三、基本用法1. 定义协程2. 运行协程3. 并发执行多个任务四、关键

MySQL的配置文件详解及实例代码

《MySQL的配置文件详解及实例代码》MySQL的配置文件是服务器运行的重要组成部分,用于设置服务器操作的各种参数,下面:本文主要介绍MySQL配置文件的相关资料,文中通过代码介绍的非常详细,需要... 目录前言一、配置文件结构1.[mysqld]2.[client]3.[mysql]4.[mysqldum

AOP编程的基本概念与idea编辑器的配合体验过程

《AOP编程的基本概念与idea编辑器的配合体验过程》文章简要介绍了AOP基础概念,包括Before/Around通知、PointCut切入点、Advice通知体、JoinPoint连接点等,说明它们... 目录BeforeAroundAdvise — 通知PointCut — 切入点Acpect — 切面

C++ STL-string类底层实现过程

《C++STL-string类底层实现过程》本文实现了一个简易的string类,涵盖动态数组存储、深拷贝机制、迭代器支持、容量调整、字符串修改、运算符重载等功能,模拟标准string核心特性,重点强... 目录实现框架一、默认成员函数1.默认构造函数2.构造函数3.拷贝构造函数(重点)4.赋值运算符重载函数

C++ vector越界问题的完整解决方案

《C++vector越界问题的完整解决方案》在C++开发中,std::vector作为最常用的动态数组容器,其便捷性与性能优势使其成为处理可变长度数据的首选,然而,数组越界访问始终是威胁程序稳定性的... 目录引言一、vector越界的底层原理与危害1.1 越界访问的本质原因1.2 越界访问的实际危害二、基

C#异步编程ConfigureAwait的使用小结

《C#异步编程ConfigureAwait的使用小结》本文介绍了异步编程在GUI和服务器端应用的优势,详细的介绍了async和await的关键作用,通过实例解析了在UI线程正确使用await.Conf... 异步编程是并发的一种形式,它有两大好处:对于面向终端用户的GUI程序,提高了响应能力对于服务器端应

Java Stream流以及常用方法操作实例

《JavaStream流以及常用方法操作实例》Stream是对Java中集合的一种增强方式,使用它可以将集合的处理过程变得更加简洁、高效和易读,:本文主要介绍JavaStream流以及常用方法... 目录一、Stream流是什么?二、stream的操作2.1、stream流创建2.2、stream的使用2.

c++日志库log4cplus快速入门小结

《c++日志库log4cplus快速入门小结》文章浏览阅读1.1w次,点赞9次,收藏44次。本文介绍Log4cplus,一种适用于C++的线程安全日志记录API,提供灵活的日志管理和配置控制。文章涵盖... 目录简介日志等级配置文件使用关于初始化使用示例总结参考资料简介log4j 用于Java,log4c