基于链表操作的飞机大战游戏

目录

飞机大战结构体游戏三部曲初始化Game_Init()绘制Game_Paint()数据更新Game_Updata()主函数按键的处理说明链表的增删操作添加敌方飞机敌方飞机移动添加子弹子弹移动

飞机大战

基于链表操作的飞机大战游戏

结构体

定义飞机大战所需结构体并全局定义结构体成员

//子弹的结构体 
struct bullet
{
    int x;
    int y;
    struct bullet* pnext;
};
// 我方飞机   
struct plane  //我方飞机的结构体
{
    int x, y;           //飞机的坐标
    bool exist;         //飞机是否存活   exist==false(0)  飞机灭亡 ;  exist=ture(1) 飞机存在
    struct bullet* bt;  //子弹
};
//敌方飞机 
struct enemy
{
    int x, y; //敌方飞机的坐标
    struct enemy* pnext; //用来保存下一个飞机的地址 
};
struct plane myPlane;        //定义我方飞机的结构体变量.
struct enemy* emy_Phead;    //敌方飞机链表的头节点. 

游戏三部曲

游戏三部曲:初始化,绘制,数据更新

初始化Game_Init()

包括加载图片和初始化游戏数据

void Game_Init()
{
    loadimage();
    myPlane.y = 700;            
    myPlane.x = rand() % 500;   //范围 0~499 
    myPlane.exist = true;       //飞机存在为true
    //初始化子弹链表的头结点
    myPlane.bt = (struct bullet *)malloc(sizeof(struct bullet));
    myPlane.bt->pnext = NULL;

    //初始化敌方飞机的头结点
    emy_Phead = (struct enemy *)malloc(sizeof(struct enemy));
    emy_Phead->pnext = NULL;
    //初始化其他变量
    begin = GetTickCount();
    t1 = GetTickCount();
    g_bk.X = 0;
    g_bk.Y = 0;
    //开局先添加两架敌方飞机
    AddEnemy();
    AddEnemy();
}
绘制Game_Paint()

包含背景图片、我方飞机,敌方飞机、子弹的绘制

void Game_Paint()
{
    //开始批量绘图
    BeginBatchDraw();
    cleardevice();  //刷新屏幕
    /*========================贴背景图片========================*/

    putimage(g_bk.X, g_bk.Y, &g_back);
    putimage(g_bk.X, g_bk.Y - 768, &g_back);

    /*========================贴我方飞机========================*/
    //判断我方的飞机是否存活,如果存活就贴飞机,否者就贴游戏失败的图片.
    if (myPlane.exist)
    {
        putimage(myPlane.x, myPlane.y, &g_plane[1], SRCAND);
        putimage(myPlane.x, myPlane.y, &g_plane[0], SRCPAINT);
    }
    else
    {
        //游戏结束
        //贴 Game Over图片
    }
    /*========================贴敌方飞机========================*/
    //遍历链表去绘制
    struct enemy* pTmp = emy_Phead->pnext; //指向第一架敌方飞机
    while (pTmp)
    {
        putimage(pTmp->x, pTmp->y, &g_enemy[1], SRCAND);
        putimage(pTmp->x, pTmp->y, &g_enemy[0], SRCPAINT);
        //要往后遍历
        pTmp = pTmp->pnext;
    }

    /*========================贴我方飞机子弹========================*/
    struct bullet* pBullet = myPlane.bt->pnext;
    while (pBullet) //贴图
    {
        putimage(pBullet->x, pBullet->y, &g_Bullet, SRCPAINT);
        pBullet = pBullet->pnext;
    }
    //结束批量绘图
    EndBatchDraw();
}
数据更新Game_Updata()

包含敌方飞机,子弹,背景的移动,我方飞机的移动有按键消息来控制不在此列

void Game_Updata()
{
    EnemyMove();    //飞机移动
    BulletMove();   //子弹移动
    BackMove();     //背景移动
}

主函数

通过不断获取end的时间来控制游戏的绘制和数据更新

通过不断获取t2的时间来控制敌方飞机的数量

通过不断获取按键消息来控制我方飞机的移动

int main()
{
    initgraph(512, 768);    /* 初始化图形库 */
    Game_Init();    /* 游戏的初始化 */
    while (1)
    {
        end = GetTickCount();   /* 获取end时间 */
        t2 = GetTickCount();    /* 获取t2时间 */
        if (kbhit())            /* 判断是否有按键消息 */
        {
            PlaneMove();    /* 玩家操作飞机移动和开火 */
        }
        if (end - begin >= 50)
        {
            Game_Paint();   /* 游戏的绘制 */
            Game_Updata();  /* 游戏的跟新 */
            begin = end;
        }
        if (t2 - t1 >= 3000)
        {
            AddEnemy();     /* 添加敌方飞机 */
            AddEnemy();     /* 添加敌方飞机 */
            t1 = t2;
        }
    }
    closegraph();   /* 关闭图形库 */
    return 0;
}

按键的处理

玩家操作实现功能

  1. 实现我方飞机的上下左右移动

  2. 实现空格添加子弹

void PlaneMove()
{
    // 获取键盘按下信息 
    char ch = getch();  //获取键盘输入
    switch (ch)     // 上 下 左  右  发射子弹
    {
    case 72:        //往上走    72 是 ↑ 键值
    case 'W':
    case 'w':
        //在里面定义变量 就要加{}
        myPlane.y -= 20;
        if (myPlane.y < 0)
        {
            myPlane.y = 0;
        }
        break;
    case 80:  //往下走
    case 'S':
    case 's':
        myPlane.y += 20;
        if (myPlane.y>668)
        {
            myPlane.y = 700;
        }
        break;
    case 75:  //左边
    case 'A':
    case 'a':
        myPlane.x -= 20;
        if (myPlane.x < 0)
        {
            myPlane.x = 0;
        }
        break;
    case 77: //右边
    case 'D':
    case 'd':
        myPlane.x += 20;
        if (myPlane.x > 437)
        {
            myPlane.x = 437;
        }
        break;
    case 32: //32是空格键
        //调用函数  发射子弹
        FireBullet();
        break;
    }
}
说明

kbbhit:检查控制台的键盘输入。

int kbhit( void );

返回值:

如果按下某个键,则kbhit返回一个非零值。否则,它返回0。

链表的增删操作

添加敌方飞机

如果ni链表操作ok的话这个敌方飞机的添加操作是完全没问题的,就一个头插

void AddEnemy()
{
    //申请新结点
    struct enemy* newnode = (struct enemy*)malloc(sizeof(struct enemy));

    newnode->x = rand() % 412;  //x坐标 随机
    newnode->y = -100;          //y坐标 固定

    //头插
    newnode->pnext = emy_Phead->pnext;
    emy_Phead->pnext = newnode;
}
敌方飞机移动

简简单单的链表遍历和删除操作,掌握基础就掌握力量

void EnemyMove()
{
    struct enemy* pTmp = emy_Phead;//指向敌方飞机的头结点
    struct enemy* pDelete;

    while (pTmp->pnext != NULL)
    {
        pTmp->pnext->y += rand() % 10; //可快可慢

        //判断飞机是否越界了
        if (pTmp->pnext->y >= 800)
        {
            //把越界的飞机销毁掉
            pDelete = pTmp->pnext;
            pTmp->pnext = pDelete->pnext;
            free(pDelete);
            pDelete = NULL;
            continue;
        }
        //往后遍历
        pTmp = pTmp->pnext;
    }
}
添加子弹

又是一个链表的头插操作,链表很重要啊

void FireBullet()
{
    //发射子弹其实就是创建结点.
    struct bullet *newbullet = (struct bullet *)malloc(sizeof(struct bullet));

    //给x和y 赋值
    newbullet->x = myPlane.x + 28; //子弹的x坐标      
    newbullet->y = myPlane.y - 10;  //子弹的 y坐标

    //连接结点 
    newbullet->pnext = myPlane.bt->pnext;
    myPlane.bt->pnext = newbullet;
}
子弹移动

这里就比较复杂了,但终究还是链表的遍历和匹配操作,注释很清楚,仔细看

  1. 通过遍历每一个子弹来实现子弹的移动

  2. 通过将每一个子弹和敌方飞机做匹配判断是否杀敌

void BulletMove()
{
    //遍历每一个子弹 让他们向上移动
    struct bullet* pPlane = myPlane.bt;         //指向子弹的头结点
    struct bullet* pDelete;                     //指向要删除的子弹
    while (pPlane->pnext != NULL)
    {
        //子弹的移动速度 都是一样的
        pPlane->pnext->y -= 10;

        //处理子弹越界
        if (pPlane->pnext->y < -50)
        {
            pDelete = pPlane->pnext;
            pPlane->pnext = pDelete->pnext;
            free(pDelete);
            pDelete = NULL;
            continue;
        }
        //处理子弹撞到飞机   子弹坐标 跟敌方飞机做比较 
        // 遍历敌方飞机 
        struct enemy *pEnemy = emy_Phead; 
        struct enemy *pDeleteEnemy;  
        //遍历敌方飞机的循环
        while (pEnemy->pnext != NULL)
        {
            if ((pPlane->pnext->x >= pEnemy->pnext->x) && pPlane->pnext->x <= (pEnemy->pnext->x + 80) && pPlane->pnext->y <= (pEnemy->pnext->y + 100))
            {
                //飞机爆炸的图片 贴上去 
                //把飞机释放掉
                pDeleteEnemy = pEnemy->pnext;
                pEnemy->pnext = pDeleteEnemy->pnext;
                free(pDeleteEnemy);
                pDeleteEnemy = NULL;

                // 把子弹释放掉 
                pDelete = pPlane->pnext;
                pPlane->pnext = pDelete->pnext;
                free(pDelete);
                pDelete = NULL;
                break;  //这个break是个坑,当子弹击中敌方某一架飞机的时候已经不需要继续遍历后面的飞机了,因为子弹已经没有了
            }
            //判断敌方飞机是不是NULL
            if (pEnemy->pnext == NULL)
            {
                break;
            }
            pEnemy = pEnemy->pnext;
        }
        //判断子弹是不是NULL
        if (pPlane->pnext == NULL)
        {
            break;  //如果子弹没了就break
        }
        pPlane = pPlane->pnext;
    }
}

 

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
使用 JavaScript 编写的记忆游戏(附源代码)   项目:JavaScript 记忆游戏(附源代码) 记忆检查游戏是一个使用 HTML5、CSS 和 JavaScript 开发的简单项目。这个游戏是关于测试你的短期 记忆技能。玩这个游戏 时,一系列图像会出现在一个盒子形状的区域中 。玩家必须找到两个相同的图像并单击它们以使它们消失。 如何运行游戏? 记忆游戏项目仅包含 HTML、CSS 和 JavaScript。谈到此游戏的功能,用户必须单击两个相同的图像才能使它们消失。 点击卡片或按下键盘键,通过 2 乘 2 旋转来重建鸟儿对,并发现隐藏在下面的图像! 如果翻开的牌面相同(一对),您就赢了,并且该对牌将从游戏中消失! 否则,卡片会自动翻面朝下,您需要重新尝试! 该游戏包含大量的 javascript 以确保游戏正常运行。 如何运行该项目? 要运行此游戏,您不需要任何类型的本地服务器,但需要浏览器。我们建议您使用现代浏览器,如 Google Chrome 和 Mozilla Firefox, 以获得更好、更优化的游戏体验。要玩游戏,首先,通过单击 memorygame-index.html 文件在浏览器中打开游戏。 演示: 该项目为国外大神项目,可以作为毕业设计的项目,也可以作为大作业项目,不用担心代码重复,设计重复等,如果需要对项目进行修改,需要具备一定基础知识。 注意:如果装有360等杀毒软件,可能会出现误报的情况,源码本身并无病毒,使用源码时可以关闭360,或者添加信任。
使用 JavaScript 编写的 Squareshooter 游戏及其源代码   项目:使用 JavaScript 编写的 Squareshooter 游戏(附源代码) 这款游戏是双人游戏。这是一款使用 JavaScript 编写的射击游戏,带有门户和强化道具。在这里,每个玩家都必须控制方形盒子(作为射手)。这款射击游戏的主要目标是射击对手玩家以求生存。当它射击对手时,它会获得一分。 游戏制作 该游戏仅使用 HTML 和 JavaScript 开发。该游戏的 PC 控制也很简单。 对于玩家 1: T:朝你上次动作的方向射击 A:向左移动 D:向右移动 W:向上移动 S:向下移动 对于玩家2: L:朝你上次移动的方向射击 左箭头:向左移动 右箭头:向右移动 向上箭头:向上移动 向下箭头:向下移动 游戏会一直进行,直到您成功射击对手或对手射击您为止。游戏得分显示在顶部。所有游戏功能均由 JavaScript 设置,而布局和其他次要功能则由 HTML 设置。 如何运行该项目? 要运行此项目,您不需要任何类型的本地服务器,但需要浏览器。我们建议您使用现代浏览器,如 Google Chrome 和 Mozilla Firefox。要运行此游戏,首先,通过单击 index.html 文件在浏览器中打开项目。 演示: 该项目为国外大神项目,可以作为毕业设计的项目,也可以作为大作业项目,不用担心代码重复,设计重复等,如果需要对项目进行修改,需要具备一定基础知识。 注意:如果装有360等杀毒软件,可能会出现误报的情况,源码本身并无病毒,使用源码时可以关闭360,或者添加信任。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值