贪吃蛇小游戏可以说是很多小伙伴的游戏启蒙之物,今天我们一起来用C语言复刻童年经典游戏贪吃蛇,在编写这个游戏之前,需要了解一些easyx图形库的知识,因为游戏的窗口就是靠此来提供。
这篇文章主要介绍编写这个游戏要实现的功能,另外我会再写一篇关于这个游戏常见的一些BUG及其解决办法,如果已经能完成部分的编写,而困于某些BUG的伙伴可以移步另一篇文章,可能会有你遇到的一些BUG。
实现贪吃蛇游戏,首先我们需要设置一个游戏窗口,利用easyx图形库,设置一个经典的640*480大小的窗口。窗口设定好了,接下来就可以定义蛇了,我们用结构体来定义蛇的类型,分析一下蛇需要用到那些数据,我们首先需要知道这个蛇有多长,便于游戏开始时蛇的长度设定,及游戏结束后蛇长的统计,那我们就用 int n 来表示蛇的长度。蛇在移动过程中是要有一个移动方向的,因此我们也需要设置一个int direction 来表示蛇移动的方向,因为方向就四个,我们干脆就把四个方向枚举出来,然后蛇的移动方向就用枚举变量来表示
//枚举方向
enum direction
{
up = 72,
down = 80,
left = 75,
right = 77
};
其次我们需要知道蛇的每一节身体的坐标,这样我们通过坐标来让蛇移动,以及把蛇绘制到窗口上,因为蛇的身体有很多节,坐标又有x , y两个值。我们就把坐标单独定义为结构体,然后身体的每一节都对应着一个坐标,我们用数组来表示身体
//定义蛇的坐标
struct coor
{
int x;
int y;
}coor;
最终蛇的定义如下
//定义蛇的数据类型
struct snake
{
int n = 3;
direction site;
coor szb[NUM];
}snake;
//其中NUM是该数组的最大值即蛇的身体最长长度,自行定义
定义完蛇的类型,接下来我们就把蛇初始化以下,写一个初始化函数, initsnake() 假设蛇刚开始长度为3, 方向向左, 坐标依次为
snake.n = 3;
snake.site = left;
snake.szb[0].x = 320;
snake.szb[0].y = 240;
snake.szb[1].x = 310;
snake.szb[1].y = 240;
snake.szb[2].x = 300;
snake.szb[2].y = 240;
初始化完成后,我们就把蛇绘制到窗口上,写一个 Drawsnake() 函数,其实就是一个循环,以蛇的长度为判定条件,从蛇头的坐标开始依次绘制,画矩形或者圆都可,这里我用的是矩形
void Drawsnake()
{
for (int i = 0; i < snake.n; i++)
{
rectangle(snake.szb[i].x, snake.szb[i].y, snake.szb[i].x+10, snake.szb[i].y+10);
}
}
绘制完成之后,接下来就要让蛇动起来了,这点很关键,如何实现让蛇动起来,分析可知,蛇移动的原理是让蛇头的坐标朝向某个方向不断的改变,身体再跟着蛇头一起改变,坐标每变换一次就重新绘制蛇的身体,因为这个过程足够快,我们将其处理的速度用Sleep()再延缓一点,就能形成我们眼睛所能识别的帧率,从而看着蛇是连续移动的。既然要不断的变换蛇头的坐标,并且重新绘制蛇的身体,那就要将这两个函数放到循环里面,蛇的移动我们编写一个Movesnake()。
int main()
{
initgraph(640, 480);
initgame();
while (ture)
{
cleardevice();
Movesnake();
Drawsnake();
Sleep(100);
}
}
那么如何让蛇的坐标不断朝着某个方向改变呢?接下来我们就仔细解决Movesnake(),首先我们要判断具体往哪个方向移动,还记得在定义蛇的时候,设置了一个方向变量site嘛,现在它派上用场了。我们在初始化的时候,将这个变量初始化向左移动,然后程序将蛇头坐标的x减少10,即向左改变了10个像素点,因为在循环里,只要site的值不变,蛇头就一直向左移动,第二节身体的坐标移动到原先蛇头的位置,其他节身体的坐标依次移动到它上一节身体的坐标处,这样就实现了蛇的移动,别忘了加上一个cleardevice()函数,清除上一次绘制的图形。
switch (snake.site)
{
case up:
snake.szb[0].y-=10;
break;
case down:
snake.szb[0].y+=10;
break;
case left:
snake.szb[0].x-=10;
break;
case right:
snake.szb[0].x+=10;
break;
}
for (int i = snake.n - 1; i > 0; i--) //将蛇的每一节身体的坐标替换成上一节身体的坐标
{
snake.szb[i].x = snake.szb[i - 1].x;
snake.szb[i].y = snake.szb[i - 1].y;
}
蛇的移动完成之后,接下来就是蛇在移动的时候,方向的改变,如果不让蛇的方向改变,那也没法玩呢,首先我们只有在按下方向键的时候蛇头的方向才会改变,那就先设置一个变量用来接收我们按下的方向键,然后再把这个方向键赋值给蛇头的方向变量,我们就编写一个Chdirection()函数来实现
void Chdirection()
{
char key;
key = _getch();
switch(key)
{
case up:
if (snake.site != down)
{
snake.site = up;
}
break;
case down:
if (snake.site != up)
{
snake.site = down;
}
break;
case left:
if (snake.site != right)
{
snake.site = left;
}
break;
case right:
if (snake.site != left)
{
snake.site = right;
}
break;
}
}
这里为什么要在每一个分支后面加上 if 判断语句呢,其实就是为了防止蛇直接就调头了,这和现实是不符的,显示中的掉头要绕圈子的。
其实,直接掉头也是可以的,还蛮有意思的,像火车一样两头开,哈哈,感兴趣可以试试。
转向的问题搞定了,但是程序怎么知道什么时候转向呢?因为这个程序是放在循环里的,我们不妨加一个判断语句,用kbhit()函数来检测玩家是否按下了按键,如果按下了就返回真值,进入转向函数,并在其内部判断该转向哪里,没有按下就为假,不进入转向函数。
int main()
{
initgraph(640, 480);
initgame();
while (ture)
{
cleardevice();
Movesnake();
Drawsnake();
Sleep(100);
if ( kbhit() )
{
Chdirection();
}
}
}
小蛇能跑起来了,接下就是吃食物了,那吃食物的效果又该如何实现呢?同样,我们要给食物定义一个数据类型,首先就是食物的坐标,其次是食物此刻的状态,是被吃了还是没被吃,状态我们就用bool变量
//定义食物的数据类型
struct food
{
int x;
int y;
bool state;
}food;
接着我们回到初始化函数处,给第一个食物的状态设定一下,定义为true,表示被吃了,为何设定为被吃了,等会解释,我们接着编写Createfood(),用来创造食物,就是给食物的坐标附上随机值,随机值就用随机值生成种子,一旦我们创建一个食物,那创建的瞬间,需要将食物的状态改成false,表示未被吃。
void Creatfood()
{
while (1)
{
food.x = rand() % 62 * 10;
food.y = rand() % 46 * 10;
if (food.x >= 20 && food.y >= 20)
{
food.state = false;
break;
}
}
}
//这里我加循环是不想让食物生成到边界点
那么什么时候生成食物呢,那当然是食物的状态为true表示被吃了的时候,这也是为什么我们要在开头处将食物的状态设置为true,就是为了开局就生成一个食物。同样要用到一个判断语句
int main()
{
initgraph(LENGTH, WIDTH);
initgame();
while (1)
{
if (food.state)
{
Creatfood();
}
cleardevice();
Movesnake();
Drawsnake();
Drawfood();
Sleep(100);
EndBatchDraw();
if (_kbhit())
{
Chdirection();
}
}
食物创建好了,现在就剩下最后一步了,那就是实现吃的过程,就编写个Eatfood(),当蛇头的坐标与食物的坐标重合时就可以吃了,然后食物的状态改为true,蛇的身长加1.
void Eatfood()
{
if (snake.szb[0].x == food.x && snake.szb[0].y == food.y)
{
snake.n++;
food.state = true;
}
}
就这样,简易的贪吃蛇就完成了,还有一个绘制食物,这个蛮简单的,各位伙伴自己实现吧。
int main()
{
initgraph(LENGTH, WIDTH);
initgame();
while (1)
{
if (food.state)
{
Creatfood();
}
cleardevice();
Movesnake();
Drawsnake();
Drawfood();
Eatfood();
Sleep(100);
if (_kbhit())
{
Chdirection();
}
}
getchar();
}