C语言实现推箱子游戏

很早就想过做点小游戏了,但是一直没有机会动手。今天闲来无事,动起手来。过程还是蛮顺利的,代码也不是非常难。今天给大家分享一下~

一、介绍

开发语言:C语言
开发工具:Dev-C++ 5.11
日期:2019年9月28日
作者:ZackSock

也不说太多多余的话了,先看一下效果图:


效果图

游戏中的人物、箱子、墙壁、球都是字符构成的。通过wasd键移动,规则的话就是推箱子的规则,也就不多说了。

二、代码实现

关于代码方面,我尽可能讲的细致。希望大家可以理解~

(1)方法列表

//主函数
void main();

//初始化一些数据
initData();

//在控制台上打印地图
drawMap();

//向上移动
moveUp();

//向左移动
moveLeft()

//向下移动
moveDown()

//向右移动
moveRight();

这几个方法都顾名思义,而且用意也非常明确,就initData可能不知道具体用处,但是没有什么大问题。唯一的问题就是,上左下右的顺序可能会逼死几个强迫症患者,哈哈。

(2)参数列表

为了方便,我把include和宏定义也放到参数列表当中

//导入函数库
#include <stdio.h>
#include <stdlib.h>
#include <conio.h>

//宏定义
#define WIDTH 8
#define HEIGHT 8

//定义地图数组,二维数组有两个维度,而地图也是二维的矩形
int map[HEIGHT][WIDTH] = {
    {0, 0, 1, 1, 1, 0, 0, 0},
    {0, 0, 1, 4, 1, 0, 0, 0},
    {0, 0, 1, 0, 1, 1, 1, 1},
    {1, 1, 1, 3, 0, 3, 4, 1},
    {1, 4, 0, 3, 2, 1, 1, 1},
    {1, 1, 1, 1, 3, 1, 0, 0},
    {0, 0, 0, 1, 4, 1, 0, 0},
    {0, 0, 0, 1, 1, 1, 0, 0} 
};

//人的位置,在二维地图中,我们可以用坐标表示一个人的位置,就好比经纬度
int x, y;

//箱子的个数,推箱子肯定要有箱子嘛。
int boxs;

这里参数不多,其中横为x,纵为y,另外这里再规定一下map的一些东西:

/**
*   0   表示空
*   1   表示墙
*   2   表示人
*   3   表示箱子
*   4   表示目的地(球)
*   5   表示已完成的箱子
*/

(3)函数具体分析

接下来我们一个一个函数来分析。

1、main函数
int main(int argc, char *argv[]) {
    char direction;     //存储键盘按的方向 
    initData();         //初始化一些数据
    
    //开始游戏的循环,这里是个死循环,每按一次按钮循环一次
    while(1){
        //每次循环的开始清除屏幕
        system("cls");
        //绘画地图
        drawMap();

        //判断,当boxs的数量0时,!0为真,然后走break跳出循环(结束游戏) 
        if(!boxs){
            break;
        }
        
        //键盘输入方向,这里使用getch,因为getch读取字符不会显示在屏幕上
        direction = getch();
        
        //用switch判断用户输入的方向
        switch(direction){
            case 'w':
                //按w时,调用向上移动函数
                moveUp();
                break;
            case 'a':
                //按a时,调用向左移动函数
                moveLeft(); 
                break;
            case 's':
                moveDown();
                break;
            case 'd':
                moveRight();
                break; 
        }
    }  
    //当跳出循环时,运行该语句,游戏结束
    printf("恭喜你完成游戏!※");
    return 0;
}

我大概说一下流程,循环外面没有什么特别的。initData()只是一些简单数据的初始化,不需要太在意。循环中大致流程如下:

  • 清除屏幕
  • 绘制地图
  • 判断游戏是否结束
  • 对用户按下的按钮进行反馈

进入循环体,先清除屏幕,再绘制地图,然后再判断游戏是否结束。可能大家对这个顺序不是很理解,这里我们先不考虑判断游戏结束的问题。我们把清屏和绘制地图合在一起,简称“重绘地图”,而游戏结束的判断先不考虑,那么流程就简化为“重绘地图 + 响应用户的操作”。简单来说就是,用户按一下按钮,我改变一下地图。

2、initData()

void initData(){
    int i, j;
    
    //加载数据时让用户等待,一般情况加载数据比较快
    printf("游戏加载中,请稍后........."); 
    
    //遍历地图中的数据
    for(i = 0; i < HEIGHT; i++){
        for(j = 0; j < WIDTH; j++){
            //遍历到2(人)时,记录人的坐标。x, y是前面定义的全局变量
            if(map[i][j] == 2){
                x = j;
                y = i;
            } 
            //遍历到3时,箱子的数目增加。boxs是前面定义的全局变量 
            if(map[i][j] == 3){
                boxs++;
            }
        }
    } 
}

这个方法很简单,就是遍历地图,然后初始化人的位置和箱子的个数。这里有一点要注意一下,就是到底内层循环是WIDTH还是外层循环是WIDTH。
地图数组

如图,在遍历过程中。外层循环控制行数,即HEIGHT。那么内层循环应该是WIDTH。

3、drawMap()

void drawMap(){
    int i, j;
    for(i = 0; i < WIDTH; i++){
        for(j = 0; j < HEIGHT; j++){
            switch(map[i][j]){
                case 0:
                    printf("  ");
                    break;
                case 1:
                    printf("■");
                    break;
                case 2:
                    printf("♀");
                    break;
                case 3:
                    printf("◆");
                    break;
                case 4:
                    printf("●");
                    break;
                case 5:
                    printf("★");
                    break; 
            }
        }
        printf("\n");
    }
}

这里也非常简单,变量map中的元素,然后通过switch判断应该输出的内容。然后内层循环每走完一次就换行。

4、moveUp()

这个函数内容有点多,想讲一下大概思路:

向上移有两种情况
1、前面为空白
    这种情况有两个步骤
    (1)将人当前的位置设置为空白(0),
    (2)再讲人前面的位置设置为人(2)
2、前面为箱子
    当前面为箱子时有三种情况
    1、箱子前面为空白
        移动人和箱子,这个操作有三个步骤
        (1)将人当前位置设置为空(0)
        (2)将箱子位置设置为人(2)
        (3)将箱子前面设置为箱子(3)
    2、箱子前面为墙
        这种情况不需要做任何操作
    3、箱子前面为终点
        这种情况有四个个步骤
        (1)将人的位置设置为空(0)
        (2)将箱子的位置设置为人(2)
        (3)将终点位置设置为★(5)
        (4)箱子boxs的数量减一
3、前面为墙
    这种情况最简单,不需要做任何操作
4、前面为终点
    我这里没有考虑太多,这种情况不做操作。(如果更换地图的话可能需要修改代码)

具体代码如下,解析我全写在注释里面:

void moveUp(){
    //定义变量存放人物上方的坐标
    int ux, uy; 
    
    //当上方没有元素时,直接return (其实人不可能在边缘)
    if(y == 0){
        return;
    }
    
    //记录上方坐标,x为横,y为纵,所有ux = x, uy = y - 1;
    ux = x;
    uy = y - 1; 
    
    //上方为已完成的箱子
    if(map[uy][ux] == 5){
        return;
    } 
    //假设上方为墙,直接return,这个和上面的判断可以合在一起,这里为了看清楚分开写 
    if(map[uy][ux] == 1){
        return;
    }
    
    //假设上方为箱子
    if(map[uy][ux] == 3){
        //判断箱子上方是否为墙 
        if(map[uy - 1][ux] == 1){
            return;
        }
        
        //判断箱子上方是否为终点
        if(map[uy - 1][ux] == 4){
            //将箱子上面内容赋值为5★ 
            map[uy - 1][ux] = 5;
            map[uy][ux] = 0;
                    
            //箱子的数目减1   
            boxs--; 
        }else{
            //移动箱子
            map[uy - 1][ux] = 3;
        }
    }
    //当上面几种return的情况都没遇到,人肯定会移动,移动操作如下
    map[y][x] = 0;
    map[uy][ux] = 2;
    //更新人的坐标
    y = uy; 
} 

这是一个方向的,其它方向要考虑的问题也和前面一样,我也就不赘述了。

6、moveLeft()

这里大致都和上面一样,就是在记录左边坐标时,应该应该是lx = x - 1。

void moveLeft(){
    //定义变量存放人物左边的坐标
    int lx, ly; 
    
    //当左边没有元素时,直接return 
    if(x == 0){
        return;
    }
    
    //记录左边坐标
    lx = x - 1;
    ly = y; 
    
    //左边为已完成方块
    if(map[ly][lx] == 5){
        return;
    } 
    
    //假设左边为墙,直接return 
    if(map[ly][lx] == 1){
        return;
    }
    
    //假设左边为箱子
    if(map[ly][lx] == 3){
        //判断箱子左边是否为墙 
        if(map[ly][lx - 1] == 1){
            return;
        }
        
        //判断箱子左边是否为球
        if(map[ly][lx - 1] == 4){
            //将箱子左边内容赋值为5★ 
            map[ly][lx - 1] = 5;
            map[ly][lx] = 0;
        
            //箱子的数目减1 
            boxs--; 
        }else{
            //移动箱子 
            map[ly][lx - 1] = 3; 
        }
    }
    map[y][x] = 0;
    map[ly][lx] = 2;
    x = lx; 
}

7、moveDown()

这里在判断边界时,判断的是 y == HEIGHT - 1。

void moveDown(){
    //定义变量存放人物下方的坐标
    int dx, dy; 
    
    //当下方没有元素时,直接return 
    if(y == HEIGHT - 1){
        return;
    }
    
    //记录下方坐标
    dx = x;
    dy = y + 1; 
    
    //下方为已完成方块
    if(map[dy][dx] == 5){
        return;
    } 
    
    //假设下方为墙,直接return 
    if(map[dy][dx] == 1){
        return;
    }
    
    //假设下方为箱子
    if(map[dy][dx] == 3){
        //判断箱子下方是否为墙 
        if(map[dy + 1][dx] == 1){
            return;
        }
        
        //判断箱子下方是否为球
        if(map[dy + 1][dx] == 4){
            //将箱子下面内容赋值为5★ 
            map[dy + 1][dx] = 5;
            map[dy][dx] = 0;
            
            //箱子的数目减1 
            boxs--; 
        }else{
            //移动箱子
            map[dy + 1][dx] = 3; 
        }
    }
    map[y][x] = 0;
    map[dy][dx] = 2;
    y = dy; 
}

8、moveRight()

这里也没什么特别说的:

void moveRight(){
    //定义变量存放人物右边的坐标
    int rx, ry; 
    
    //当右边没有元素时,直接return 
    if(x == WIDTH - 1){
        return;
    }
    
    //记录右边坐标
    rx = x + 1;
    ry = y; 
    
    //右边为已完成方块
    if(map[ry][rx] == 5){
        return;
    } 
    
    //假设右边为墙,直接return 
    if(map[ry][rx] == 1){
        return;
    }
    
    //假设右边为箱子
    if(map[ry][rx] == 3){
        //判断箱子右边是否为墙 
        if(map[ry][rx + 1] == 1){
            return;
        }
        
        //判断箱子左边是否为球
        if(map[ry][rx + 1] == 4){
            //将箱子右边内容赋值为5★ 
            map[ry][rx + 1] = 5;
            map[ry][rx] = 0;
            
            //箱子的数目减1 
            boxs--; 
        }else{
            //移动箱子 
            map[ry][rx + 1] = 3; 
        }
    }
    map[y][x] = 0;
    map[ry][rx] = 2;
    x = rx; 
}

三、总结

现在再回顾开始的运行步骤

  • 清除屏幕
  • 绘制地图
  • 判断游戏是否结束
  • 对用户按下的按钮进行反馈

这里把判断游戏是否结束放到了重绘图像后面,因为在对用户进行反馈的时候只是改变了map中的数据,实际上最后一个箱子推到终点的图像还没有显示出来,所以要在重绘之后再判断是否结束游戏。

代码有很多冗余的地方,一方面是想大家更好的理解,还有一方面出于懒。哈哈,代码运行起来没有问题,源码和源程序我会上传,有兴趣的可以下下来,或者直接复制代码运行也是没问题的。
百度云连接如下:
链接:https://pan.baidu.com/s/1pwEKt3VTKmssDgU6dLx0pg 提取码:4o9v

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,456评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,370评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,337评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,583评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,596评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,572评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,936评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,595评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,850评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,601评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,685评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,371评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,951评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,934评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,167评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,636评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,411评论 2 342

推荐阅读更多精彩内容