你问我为什么要用c写贪吃蛇游戏?一切都要从那次作业讲起。。。

更多精彩文章请关注公众号『大海的BLOG』

背景:

大一上半年,C程序设计老师布置了一个程序设计作业,于是诞生了这篇代码。

IDE:

VisualStudio2017社区版

字符集:

使用多字节字符集

辅助工具:

EasyX图形库2018春分版

PS:

1、因为当时想得高分就以为搞点花里胡哨的就不错(哈哈最后混了个优,乐死我了),所以我编写了账号注册、登录系统、调用了多媒体接口以登录游戏时播放背景音乐、高分榜系统以及附带的升序降序排行函数:

​ ①账号注册、登录系统、高分榜系统的数据全部分别保存在默认添加创建的Users.dat,scores.dat里,所以要借鉴的朋友需要手动在IDE里添加这两个文件(对应名称可以自己改,但一定要对应到代码里,否则会fp指针调用文件出错);

​ ②关于背景音乐:我添加的背景音乐的资源就不放到帖子上了(名字叫silver city,好听,哈哈哈),所以要借鉴源码的朋友需要自己了解一下“如何添加MP3格式的音乐到代码”的问题,然后对代码第328行进行改写;

2、操作问题:因为我喜欢玩FPS游戏,所以定义的上下左右是wsad(小写,一定要把输入法切换为小写英文状态,否则蛇死亡),所以不习惯的朋友可以重新定义为键盘方向键(需要了解一下键盘虚拟键值,然后改对应的switch语句,代码量应该还是挺大的)。

3、EasyX图形库:(这段是复制我设计报告的23333333)在我决定用EasyX库的图形函数来绘制地图之前,对于这个模块的编写有很多问题:譬如单用gotoxy()函数实现绘制地图的话,一片黑白,惨不忍睹;而且也会成几何倍的增加代码量:尤其是for循环语句和printf语句会多不胜数。从而就会导致不仅开发人员降低对于代码的可读性,而且从头至尾的黑白界面无疑会劝退很多玩家,降低可玩性。在我用EasyX来绘制地图后,测试的过程只能用两个字来形容:真香!EasyX的一个moveto()配合一个for循环代替了多不胜数的gotoxy()函数和for循环。这不仅大大缩减了代码量,而且图形编程的优点“多彩化”也得以体现。在调试过程中我对for循环进行了优化:一个进行边界的上下绘制,另一个负责左右的绘制,实现了不仅减少了代码量而且没有破坏不影响可读性的前提。

4、应程序设计老师要求,基本上能达到每个功能、每个函数甚至每一句都有一条注释。。。(想想都要吐了)


放代码吧:

#include                         //C语言标准输入输出头文件
#include                 // easyx图形库头文件
#include                         //获取数据头文件(getch()和kbhit())
#include                 //使用其中的Sleep函数控制循环的时间,system("cls")函数来清屏
#include                         //调用了srand()随机函数
#include                         //使用当前时钟做随机种子
#include                 //多媒体设备接口头文件
#pragma comment (lib,"winmm.lib")

//首先进行一堆宏定义

//定义地图大小
#define frame_height 30 
#define frame_width 30
//定义操作键,小写的wasd控制上左下右
#define UP 'w' 
#define DOWN 's'
#define LEFT 'a'
#define RIGHT 'd'
//控制台的点是单位点,不考虑大小,我们用的easyx绘制像素点为了将单位点在图形界面上放大为一个正方形,系数转换×16可以把一个单位点放大成正方形。16可以是任意数,相应的会放大或缩小所有的像素点。
#define SIZE 16

#define N 4

//声明fp是指针,用来指向FILE类型的对象。
FILE *fp;

int i, j, k, m, n, s[N];

char s1[4], s2[4];
//蛇的初始方向
char ch = UP;
//定义一个全局变量,来判断蛇是否成长,成长则速度加快。grow值也相应增加。
int grow = 0;



//食物的坐标
struct Food {
        int x;
        int y;
}food;

//蛇的信息:snake[0]是蛇头,len为蛇的长度,speed蛇的速度
struct Snake {
        int x[50];
        int y[50];
        int len;
        int speed;
}snake;

//用于储存用户账号密码
typedef struct The_users
{
        char id[11]; //账号
        char pwd[20];//密码
}users;
users a, b;//新建结构体成员变量,用来登陆与注册


//此处声明变量

void init_map(void);
void update_food(void);
void move_snake(void);
int is_alive(void);
void get_speed(void);
void gotoxy(int x, int y);
void redraw_map(void);
void menu();
void registers();
void Login();
//void Create_File();
void score();
int compInc(const void *a, const void *b);
int compDec(const void *a, const void *b);


//主函数位置
int main()
{
        initgraph(640, 480);        //创建绘图窗口,长640.
        menu();                                        //调用菜单函数
        score();                                //调用得分函数
        return 0;
}


//再次绘制地图(根据宏定义中的宽和高)
void redraw_map(void)
{
        for (j = 0; j < frame_width; j++)                        //该for循环用于绘制地图的上下边界,根据width宽来从左往右、先上后下依次绘制。
        {
                moveto(j*SIZE, 0);                                                //将画笔移动到(j*SIZE,0)的位置开始进行下面的绘图,另外(j*SIZE)是将单位点放大SIZE(16)倍
                setfillcolor(BLUE);                                                //定义上边界为蓝色
                fillrectangle(j*SIZE, 0, j*SIZE + SIZE, SIZE);        //绘制矩形
                moveto(j*SIZE, (frame_height - 1)*SIZE);//再次移动画笔
                setfillcolor(BLUE);                                                //定义下边界为蓝色
                fillrectangle(j*SIZE, (frame_height - 1)*SIZE, j*SIZE + SIZE, (frame_height - 1)*SIZE + SIZE);//绘制矩形
        }
        for (i = 1; i < frame_height - 1; i++)//次循环绘制地图左右剩余的28个矩形方块,先左后右,从上至下!
        {
                moveto(0, i*SIZE);                                //画笔移动到(0,i*SIZE)
                setfillcolor(BLUE);                                //定义左边界为蓝色
                fillrectangle(0, i*SIZE, 0 + SIZE, i*SIZE + SIZE);
                moveto((frame_width - 1)*SIZE, i*SIZE);//再次移动画笔
                setfillcolor(BLUE);                                        //定义右边界为蓝色
                fillrectangle((frame_width - 1)*SIZE, i*SIZE, (frame_width - 1)*SIZE + SIZE, i*SIZE + SIZE); //绘制矩形
        }
}

//然后初始化地图内容
void init_map(void)
{
        //初始化食物
        srand((unsigned int)time(NULL));//用srand函数集结合time.h头文件中的time来生成随机数
        food.x = rand() % (frame_height - 2) + 1;//在1~(frameheight-2)之间随机生成一个食物的横坐标
        food.y = rand() % (frame_width - 2) + 1;//在1~(framewidth -2)之间随机生成一个食物的纵坐标
        moveto(food.y*SIZE, food.x*SIZE);                //将画笔移动到该随机坐标
        setfillcolor(RED);                                                //定义食物为红色
        fillcircle(food.y *SIZE + SIZE / 2, food.x*SIZE + SIZE / 2, SIZE / 2);                //定义食物形状为以(food.y *SIZE + SIZE / 2, food.x*SIZE + SIZE / 2)为圆心,SIZE/2为半径的圆形

        //初始化蛇
        snake.x[0] = (frame_height) / 2;//初始化蛇头横坐标位置
        snake.y[0] = (frame_width) / 2;//初始化蛇头纵坐标位置
        moveto(snake.y[0] * SIZE, snake.x[0] * SIZE);//画笔移动到蛇头坐标
        setfillcolor(GREEN);                        //定义蛇的初始颜色为绿色
        fillcircle(snake.y[0] * SIZE + SIZE / 2, snake.x[0] * SIZE + SIZE / 2, SIZE / 2);//画蛇
        snake.len = 3;                                                //初始化蛇的节数为3节
        snake.speed = 200;                                        //初始化蛇的速度
        for (k = 1; k < snake.len; k++)                //用for循环画出蛇剩余的节数
        {
                snake.x[k] = snake.x[k - 1] + 1;
                snake.y[k] = snake.y[k - 1];
                moveto(snake.y[k] * SIZE, snake.x[k] * SIZE);
                setfillcolor(GREEN);
                fillcircle(snake.y[k] * SIZE + SIZE / 2, snake.x[k] * SIZE + SIZE / 2, SIZE / 2);
        }
}

//生成食物
void update_food()
{
        if (snake.x[0] == food.x&&snake.y[0] == food.y)//首先保证每次随机的食物不能刚好随机到蛇头所在坐标
        {
                food.x = rand() % (frame_height - 2) + 1;
                food.y = rand() % (frame_width - 2) + 1;
                for (k = 1; k < snake.len; k++)
                {
                        if (snake.x[k] == food.x&&snake.y[k] == food.y)//判断蛇头坐标等于随机食物的坐标 即:吃到食物
                        {
                                food.x = rand() % (frame_height - 2) + 1;//再次随机生成食物
                                food.y = rand() % (frame_width - 2) + 1;
                        }
                }
                moveto(food.y*SIZE, food.x*SIZE);//移动画笔到随机食物的坐标
                setfillcolor(RED);                                //定义食物颜色为红色
                fillcircle(food.y*SIZE + SIZE / 2, food.x*SIZE + SIZE / 2, SIZE / 2);                //定义食物为圆形
                snake.len++;                                        //蛇的长度加一
                grow = 1;                                                //给全局变量grow赋值1
        }
}

//蛇的移动
void move_snake()
{
        if (_kbhit())//kbhit判断键盘输入的虚拟键值
                ch = _getch();//将方向键值表现为对应的方向
        if (!grow)//此时grow值若为零则执行if语句,不为零则不执行
        {
                moveto(snake.y[snake.len - 1] * SIZE, snake.x[snake.len - 1] * SIZE);
                setfillcolor(BLACK);
                solidrectangle(snake.y[snake.len - 1] * SIZE, snake.x[snake.len - 1] * SIZE, snake.y[snake.len - 1] * SIZE + SIZE, snake.x[snake.len - 1] * SIZE + SIZE);
        }
        for (k = snake.len - 1; k > 0; k--)
        {
                snake.x[k] = snake.x[k - 1];
                snake.y[k] = snake.y[k - 1];
        }
        switch (ch)
        {
        case UP:  snake.x[0]--; break;
        case DOWN: snake.x[0]++; break;
        case LEFT:  snake.y[0]--; break;
        case RIGHT:  snake.y[0]++; break;
        default:  break;                                        //其他按键直接导致死亡
        }
        moveto(snake.y[0] * SIZE, snake.x[0] * SIZE);
        setfillcolor(GREEN);
        fillcircle(snake.y[0] * SIZE + SIZE / 2, snake.x[0] * SIZE + SIZE / 2, SIZE / 2);
        grow = 0;
}

//判断是否存活
int is_alive(void)
{
        if (snake.x[0] == 0 || snake.x[0] == frame_height - 1 || snake.y[0] == frame_width - 1 || snake.y[0] == 0)//判断是否撞墙
                return 0;                //死
        for (k = 1; k < snake.len; k++)
                if (snake.x[k] == snake.x[0] && snake.y[k] == snake.y[0])//判断蛇头是否撞到蛇的身体
                        return 0;        //死
        return 1;                        //活
}

//通过节数来提升速度
void get_speed(void)
{
        if (snake.len <= 6)
                snake.speed = 200;
        else if (snake.len <= 10)
                snake.speed = 100;
        else if (snake.len <= 20)
                snake.speed = 50;
        else if (snake.len <= 30)
                snake.speed = 30;
        else snake.speed = 20;
}

//移动光标
void gotoxy(int x, int y)
{
        HANDLE hout;
        COORD cor;
        hout = GetStdHandle(STD_OUTPUT_HANDLE);
        cor.X = y;
        cor.Y = x;
        SetConsoleCursorPosition(hout, cor);
}

//进入界面
void menu()
{
        InputBox(s1, 4, "输入数字以选择目的:\n1.开始游戏\n2.高分榜\n3.退出游戏\n游戏说明:wasd控制上左下右(小写)", "疯狂的蛇——海制作", NULL, 0, 0, false);
        int x, y[3];
        sscanf_s(s1, "%d", &x);//将用户输入转化为数字
        if (x == 1)
        {
                InputBox(s2, 4, "输入数字以选择目的:\n1.注册账号\n2.登陆游戏", "疯狂的蛇——海制作");
                sscanf_s(s2, "%d", &x);
                if (x == 1) {
                        registers();
                        Login();
                }
                else if (x == 2) {
                        Login();
                }
                init_map(); //初始化地图
                while (1)
                {
                        update_food();
                        get_speed();
                        move_snake();
                        redraw_map();
                        Sleep(snake.speed);
                        if (!(is_alive()))
                                break;
                }
                closegraph();                                                //关闭游戏窗口
                printf("哈哈小辣鸡,游戏结束!\n");
                printf("你的得分:%d", snake.len - 3);//得分=(蛇死亡前节数len)-(蛇的初始节数3)
                _getch();
        }
        else if (x == 2)
        {
                closegraph();                                                //关闭游戏窗口
                printf("                    排行榜\n");
                fopen_s(&fp, "scores.dat", "r");
                for (x = 0; x < N - 1; x++) {
                        fscanf_s(fp, "%d", &y[x]);
                        printf("%d\n", y[x]);
                }
                fclose(fp);
                _getch();

        }
        else//输入3或其他除了1、2任意键均退出游戏
        {
                closegraph();
        }
}

//注册系统
void registers()
{
        fopen_s(&fp, "Users.dat", "r");
        fscanf_s(fp, "%s%s", b.id, sizeof(b.id), b.pwd, sizeof(b.pwd));
        InputBox(a.id, 11, "请输入你的账号:", "疯狂的蛇——海制作");
        while (1)
        {
                if (strcmp(a.id, b.id) != 0)//如果两串字符串不相等
                {
                        if (!feof(fp))//如果未至文件末尾,它的工作原理是,站在光标所在位置,向后看看还有没有字符。如果有,返回0;如果没有,返回非0。它并不会读取相关信息,只是查看光标后是否还有内容。
                        {
                                fscanf_s(fp, "%s%s", b.id, sizeof(b.id), b.pwd, sizeof(b.pwd));
                        }
                        else break;
                }
                else
                {
                        outtextxy(220, 200, "此用户名已被注册");
                        fclose(fp);
                        _getch();
                        exit(0);
                }
        }
        fclose(fp);
        InputBox(a.pwd, 10, "请输入你的密码:", "疯狂的蛇——海制作");
        fopen_s(&fp, "Users.dat", "a");
        fprintf_s(fp, "%s %s\n", a.id, a.pwd);
        outtextxy(220, 200, "奥利给!恭喜你!账号注册成功!");
        fclose(fp);
}

//登陆账号密码,登录时播放bgm嘿嘿sao起来
void Login()
{
        mciSendString("open SC.mp3 alias BIGOcean", 0, 0, 0);
        mciSendString("play BIGOcean repeat", 0, 0, 0);
        fopen_s(&fp, "Users.dat", "r");
        fscanf_s(fp, "%s%s", b.id, sizeof(b.id), b.pwd, sizeof(b.pwd));
        InputBox(a.id, 11, "请输入账号", "疯狂的蛇——海制作");
        while (1)
        {
                if (strcmp(a.id, b.id) == 0) break;//如果找到了这个用户名
                else
                {
                        if (!feof(fp))//如果文件未读完
                                fscanf_s(fp, "%s%s", b.id, sizeof(b.id), b.pwd, sizeof(b.pwd));
                        else
                        {
                                outtextxy(220, 200, "此用户名不存在!");
                                fclose(fp);
                                _getch();
                                exit(0);
                        }
                }
        }
        InputBox(a.pwd, 20, "请输入密码", "疯狂的蛇——海制作");
        if (strcmp(a.pwd, b.pwd) == 0)//如果密码匹配
        {
                fclose(fp);
                outtextxy(250, 200, "登陆成功!奥利给!");
                initgraph(640, 480);
        }
        else
        {
                outtextxy(220, 200, "STFU!密码不正确");
                _getch();
                exit(0);
        }
}

//分数系统
void score()
{
        fopen_s(&fp, "scores.dat", "r");
        for (n = 0; n < N - 1; n++)
        {
                fscanf_s(fp, "%d", &m);
                s[n] = m;
        }
        s[N - 1] = snake.len - 3;
        qsort(s, N, sizeof(s[0]), compDec);
        fclose(fp);
        fopen_s(&fp, "scores.dat", "w");
        for (n = 0; n < N - 1; n++)
        {
                fprintf_s(fp, "%d\n", s[n]);
        }
        fclose(fp);
}

//分数的排序
//升序排序
int compInc(const void *a, const void *b)
{
        return *(int *)a - *(int *)b;
}
//降序排序
int compDec(const void *a, const void *b) {
        return *(int *)b - *(int *)a;
}

更多精彩文章请关注公众号『大海的BLOG』


学习中...