显示界面如上图所示
自己找的背景和飞机素材,先将素材奉上.
接下来我先简单分析一下这个单机游戏的运行逻辑:
就像显示界面所显示的那样,我们想要实现的是自己的飞机在发射子弹(子弹在上图没显示),然后当子弹射到敌方飞机,这里设置了两种类型的飞机,如果读者想定义更多类型的,直接添加属性就可以.当子弹射到飞机的时候,飞机的血量减少,就像CS一样,并不一定是一发子弹就死亡,读者可以根据自己喜欢去设置,博主这里设置大飞机的血量是3发子弹,而小飞机的血量是1发子弹,也就是说,当有3发子弹射到大飞机的时候,大飞机就消失,一发子弹小飞机消失.至于其他的动态效果,博主这里暂时不做介绍.具体往下看:
面向对象开发,每一个对象我们都需要给他设置相对应的对象属性,在语言当中,我们将其设置成结构体就可以:
enum My
{WIDTH = 500,HIGHT = 700,BULLET_NUM = 100,//玩家子弹数量ENEMY_NUM = 10,//敌机的数量BIG,SMALL,
};
struct Plane
{int x;int y;bool live;//是否存活int width;int hight;int hp;int type;//敌机类型 big small}player,bull[BULLET_NUM],enemy[ENEMY_NUM];
首先设定了一个联合,用于设置弹出窗口的基本参数和其他属性.
上面是不是把敌方飞机,自己飞机和自己发射的子弹都定义成了一个结构体.下面我们就可以直接用了.
(x,y)代表的是对象在窗口当中的坐标,width和hight是对象自身的尺寸,而hp是血量,type是博主自己定义了两种敌机类型.布尔类型live判断飞机是否存活状态.
第一步我们应该是要先创建图形窗口
/*测试
circle(50, 50, 50);//画空心圆
setfillcolor(GREEN);//圆的颜色
fillcircle(100, 100, 50);///实心⚪*///创建图形窗口initgraph(WIDTH, HIGHT,SHOWCONSOLE);//宽长
initgraph()函数直接用
第二步加载窗口里面使用到的图片 //无论什么时候用,我们都要进行图片加载的,因此此处我们将其设定到一个函数下,方便之后也可以随时添加和删除.
//把图片加载进程序
IMAGE bk;//背景图
//保存玩家图片
IMAGE img_role;
//保存子弹
IMAGE img_bull;
//
IMAGE img_enemy[4];
void loadImg()
{//加载图片loadimage(&bk, "D:/program/飞机大战/images/bk.png");//加载玩家图片loadimage(&img_role, "D:/program/飞机大战/images/fly.png");//加载子弹图片loadimage(&img_bull, "D:/program/飞机大战/images/子弹.jpg");//加载敌机的图片loadimage(&img_enemy[0], "D:/program/飞机大战/images/enemy1.png");loadimage(&img_enemy[1], "D:/program/飞机大战/images/enemy2.png");//loadimage(&img_enemy[2], "D:/program/飞机大战/images/enemy3.jpg");//loadimage(&img_enemy[3], "D:/program/飞机大战/images/enemy4.jpg");
}
IMAGE 是easyx里面的一个参数,直接用
此案例当中,需要我们加载的图片有背景图,自己飞机图,子弹图,敌方飞机1和敌方飞机2,如果读者也自己添加第三台和第四台敌机,也可以直接添加,不过要注意,当一种对象,存在多个类型的时候,要采用数组将其存储,方便之后进行使用.
第三步,要干嘛?设置好窗口了,加载完图片了,是不是要把图片添加进去了?
那图片添加到哪里去?怎么添加?哪一种图片应该添加在哪里呢?
这里我们是不是首先要初始化一下图片的基本属性?of course,买衣服我总的知道我适合穿多大尺码的衣服吧?
void gameinit()//初始化数据
{player.x = WIDTH / 2;player.y = HIGHT - 120;player.live = true;//初始化子弹for (int i = 0; i < BULLET_NUM; i++){bull[i].x = 0;bull[i].y = 0;bull[i].live = false;}//初始化敌机for (int i = 0; i < ENEMY_NUM; i++){enemy[i].live = false;enemyHP(i);}
}
想想我们一共有几个对象,去除背景图片不需要我们在这里初始化之外,另外就是自己飞机,敌方飞机和子弹三个对象.那么我们就定义一个函数,用于初始化这些对象的第一属性.
在初始化敌方飞机的时候,博主之前把敌方飞机设置有两种类型,那么这两种类型飞机的基本属性一定有一些是不一样的呀,要不然怎么叫做2呢?
博主这里主要用飞机的大小和血量来区分这两种飞机.具体看enemyHP()函数
void enemyHP(int i)
{int flag = rand() % 10;if (flag>=0 && flag<=2)//0-9{enemy[i].type = BIG;enemy[i].hp = 3;enemy[i].width = 100;enemy[i].hight = 100;}else{enemy[i].type = SMALL;enemy[i].hp = 1;enemy[i].width = 50;enemy[i].hight = 50;}}
在这个函数当中,我们要注意,大飞机和小飞机出现的次数是不是正常情况下是不一样的?比如小飞机出现了5次,而大飞机才出现一次这样,当然敌飞机出现肯定不是我们可以控制的,它肯定是随机出现的,所以这里我们采用一个随机范围,如果在0-2之间就出现大飞机,3-9之间小飞机,这样是不是小飞机的概率要高于大飞机.其余的进入条件之内直接初始化就可以了.
第四步,上面我们把全部图片都加载好,然后对象都进行了初始化,接下来我们要干嘛?
那肯定是要将这些图片绘制到对应的窗口坐标上了!!!!!!!
void gameDraw()
{loadImg();//把背景贴在窗口上putimage(0, 0, &bk);putimage(player.x, player.y, &img_role,SRCINVERT);for (int i = 0; i < BULLET_NUM; i++){if (bull[i].live){putimage(bull[i].x, bull[i].y, &img_bull, SRCINVERT);}}//绘制敌机for (int i = 0; i < ENEMY_NUM; i++){if (enemy[i].live){if(enemy[i].type==SMALL)putimage(enemy[i].x, enemy[i].y, &img_enemy[0], SRCINVERT);//x和y是飞机在窗口中的坐标elseputimage(enemy[i].x, enemy[i].y, &img_enemy[1], SRCINVERT);}}}
绘制背景,绘制自己飞机,绘制子弹,绘制敌方飞机都采用putimage()函数就可以了
需要注意的有可能就是对于敌方飞机和子弹只有live是true的时候才绘制,而对于敌方飞机还区分了两种类型的飞机,加一个if-else输出对应的敌方飞机就可以了;
第五步,第四步我们绘制的图片是个啥,是初始化的位置,是一个静态的图,我们要实现的动态图,那接下来我们应该干嘛????????
当然是让子弹,自己飞机和敌方飞机都动起来了,也就是创建动画
首先让自己飞机动起来
void playermove(int speed)
{
#if 0if (_kbhit) {//有两种方式//1,getch() 阻塞函数,和scanf一样,如果没有输入,就会卡住程序,一直等待输入,//这个函数不是c语言标准函数,需要头文件<conio.h>char key = _getch();switch (key){case 'w':case 'W':player.y -= speed;break;case 's':case 'S':player.y += speed;break;case 'a':case 'A':player.x -= speed;break;case 'd':case 'D':player.x += speed;break;default:break;}}
}
#elif 1//2.使用Windows函数获取键盘输入//非阻塞函数,特别流畅//如果用字母,必须要用大写,大写可以识别大小写if (GetAsyncKeyState(VK_UP) || GetAsyncKeyState('W')){if(player.y > 0)player.y -= speed;}if (GetAsyncKeyState(VK_DOWN) || GetAsyncKeyState('S')){if (player.y < HIGHT-78)//78是飞机的高度player.y += speed;}if (GetAsyncKeyState(VK_LEFT) || GetAsyncKeyState('A')){if(player.x+59>0)// 119/2 = 59是控制飞机子弹在左右边界的时候可以打到敌机player.x -= speed;}if (GetAsyncKeyState(VK_RIGHT) || GetAsyncKeyState('D')){if (player.x-59 < WIDTH-119)//119是飞机的宽度player.x += speed;}
#endif // 0if (GetAsyncKeyState(VK_SPACE) && Timer(100,1))//{//创建一个子弹createBullet();}}
这里尝试了两种让自己飞机动起来的方法,第一种是_kbhit,如果按下键盘,判断按下的是什么键,如果wsad当中的一个则执行对应的操作,但是这种方法运行之后,你会发现很卡,就是一个帧当中只能运行一个动作,不能运行右上这样的操作,关键是很卡
所以我们用了第二种方案非阻塞函数GetAsyncKeyState(),分别对键盘当中的上下左右和英文字符当中的wasd进行case分析,其他不变,此处要注意的有可能就是要控制自己的飞机运行,只能在窗口范围内运行,为其设定边界就可以.
最后还加了一个和子弹相关的函数,如果说按下空格键了,就要发射子弹了,同时这里结合了一个定时器的函数,用于控制子弹发射的速度100ms,不要再100ms内发射两颗子弹,防止出现子弹重叠的现象.
bool Timer(int ms, int id)//定时器
{static DWORD t[10];if (clock() - t[id] > ms){t[id] = clock();return true;}return false;
}
定义了一个无符号长整形的时间数组,是因为本案例当中有多个函数都用到了定时器函数.
自己飞机动起来,接下来就很明显了,让子弹跟着发射,对不对????也就是进入createBullet()函数
void createBullet()
{for (int i = 0; i < BULLET_NUM; i++){if (!bull[i].live){bull[i].x = player.x+59;bull[i].y = player.y;bull[i].live = true;break;}}
}
刚开始子弹都是的状态都是false,然后条件正确后,初始化了子弹的第一个位置发射出来,然后将子弹的状态改为true.跳出这一个,执行下一个,要注意此时只是静态化绘制了子弹,还没有动起来,那么接下来就让子弹垂直向上运行就好.
void Bullmove(int speed)
{for (int i = 0; i < BULLET_NUM; i++){if (bull[i].live){bull[i].y -=speed;if (bull[i].y < 0){bull[i].live = false;}}}
}
如果子弹到了最上方了,改变子弹的状态为false,继续循环
自己飞机和子弹都结束了,接下来就是敌方飞机了
//产生敌机
void createenemy()
{for (int i = 0; i < ENEMY_NUM; i++){if (!enemy[i].live){enemy[i].live = true;enemy[i].x = rand()%(WIDTH-75);enemy[i].y = 0;enemyHP(i);printf("pos(%d,%d) %d %d\n", enemy[i].x, enemy[i].y, enemy[i].live, enemy[i].hp);break;}}
}
在窗口的最上面随机产生敌机,敌机也要动的.是不是参考子弹的就可以了,如下
//敌机的移动
void enemymove(int speed)
{for (int i = 0; i < ENEMY_NUM; i++){if (enemy[i].live){enemy[i].y += speed;if (enemy[i].y > HIGHT){enemy[i].live = false;}}}
}
最后就是打飞机,当子弹到了敌方飞机的范围内是,敌方飞机就消失
//开始打飞机
void playplane()
{for (int i = 0; i < ENEMY_NUM; i++){if (!enemy[i].live)continue;for (int k = 0; k < BULLET_NUM; k++){if (!bull[k].live)continue;if (bull[k].x > enemy[i].x && bull[k].x<enemy[i].x + enemy[i].width&& bull[k].y>enemy[i].y && bull[k].y < enemy[i].y + enemy[i].hight)//子弹在飞机的界面内//则代表子弹打住了飞机{bull[k].live = false;enemy[i].hp--;}}if (enemy[i].hp == 0){enemy[i].live = false;}}
}
if (bull[k].x > enemy[i].x && bull[k].x<enemy[i].x + enemy[i].width&& bull[k].y>enemy[i].y && bull[k].y < enemy[i].y + enemy[i].hight)//子弹在飞机的界面内//则代表子弹打住了飞机
这是个啥?意思是当子弹到了敌方飞机那个矩形图之内,条件就正确
这里要注意大飞机血量是3,小飞机血量是1,一颗子弹一格血量.当血量为0,则飞机死亡
最后一步,上主函数
int main()
{/*测试
circle(50, 50, 50);//画空心圆
setfillcolor(GREEN);//圆的颜色
fillcircle(100, 100, 50);///实心⚪*///创建图形窗口initgraph(WIDTH, HIGHT,SHOWCONSOLE);//宽长gameinit();//双缓冲绘图BeginBatchDraw();while (1){gameDraw();FlushBatchDraw();playermove(5);Bullmove(5);//防止飞机全部一起出来if (Timer(500,0)){//创建一个飞机createenemy();}if (Timer(20, 2)){enemymove(1);}playplane();}EndBatchDraw();return 0;
}
注意这里的
//双缓冲绘图BeginBatchDraw();EndBatchDraw();
是easyx插件里面的一个双缓冲绘图,防止图片在窗口当中出现闪屏
总代码:
#include<stdio.h>
//图形库,帮助我们新手,快速入门图形编程easyx
#include<graphics.h>
#include<conio.h>
#include<time.h>
enum My
{WIDTH = 500,HIGHT = 700,BULLET_NUM = 100,//玩家子弹数量ENEMY_NUM = 10,//敌机的数量BIG,SMALL,
};
struct Plane
{int x;int y;bool live;//是否存活int width;int hight;int hp;int type;//敌机类型 big small}player,bull[BULLET_NUM],enemy[ENEMY_NUM];//把图片加载进程序
IMAGE bk;//背景图
//保存玩家图片
IMAGE img_role;
//保存子弹
IMAGE img_bull;
//
IMAGE img_enemy[4];void loadImg()
{//加载图片loadimage(&bk, "D:/program/飞机大战/images/bk.png");//加载玩家图片loadimage(&img_role, "D:/program/飞机大战/images/fly.png");//加载子弹图片loadimage(&img_bull, "D:/program/飞机大战/images/子弹.jpg");//加载敌机的图片loadimage(&img_enemy[0], "D:/program/飞机大战/images/enemy1.png");loadimage(&img_enemy[1], "D:/program/飞机大战/images/enemy2.png");//loadimage(&img_enemy[2], "D:/program/飞机大战/images/enemy3.jpg");//loadimage(&img_enemy[3], "D:/program/飞机大战/images/enemy4.jpg");
}
bool Timer(int ms, int id)//定时器
{static DWORD t[10];if (clock() - t[id] > ms){t[id] = clock();return true;}return false;
}
void enemyHP(int i)
{int flag = rand() % 10;if (flag>=0 && flag<=2)//0-9{enemy[i].type = BIG;enemy[i].hp = 3;enemy[i].width = 100;enemy[i].hight = 100;}else{enemy[i].type = SMALL;enemy[i].hp = 1;enemy[i].width = 50;enemy[i].hight = 50;}}void gameinit()//初始化数据
{player.x = WIDTH / 2;player.y = HIGHT - 120;player.live = true;//初始化子弹for (int i = 0; i < BULLET_NUM; i++){bull[i].x = 0;bull[i].y = 0;bull[i].live = false;}//初始化敌机for (int i = 0; i < ENEMY_NUM; i++){enemy[i].live = false;enemyHP(i);}
}
//游戏绘制函数
void gameDraw()
{loadImg();//把背景贴在窗口上putimage(0, 0, &bk);putimage(player.x, player.y, &img_role,SRCINVERT);for (int i = 0; i < BULLET_NUM; i++){if (bull[i].live){putimage(bull[i].x, bull[i].y, &img_bull, SRCINVERT);}}//绘制敌机for (int i = 0; i < ENEMY_NUM; i++){if (enemy[i].live){if(enemy[i].type==SMALL)putimage(enemy[i].x, enemy[i].y, &img_enemy[0], SRCINVERT);//x和y是飞机在窗口中的坐标elseputimage(enemy[i].x, enemy[i].y, &img_enemy[1], SRCINVERT);}}}void createBullet()
{for (int i = 0; i < BULLET_NUM; i++){if (!bull[i].live){bull[i].x = player.x+59;bull[i].y = player.y;bull[i].live = true;break;}}
}
void Bullmove(int speed)
{for (int i = 0; i < BULLET_NUM; i++){if (bull[i].live){bull[i].y -=speed;if (bull[i].y < 0){bull[i].live = false;}}}
}//角色移动,获取键盘信息,上下左右
void playermove(int speed)
{
#if 0if (_kbhit) {//有两种方式//1,getch() 阻塞函数,和scanf一样,如果没有输入,就会卡住程序,一直等待输入,//这个函数不是c语言标准函数,需要头文件<conio.h>char key = _getch();switch (key){case 'w':case 'W':player.y -= speed;break;case 's':case 'S':player.y += speed;break;case 'a':case 'A':player.x -= speed;break;case 'd':case 'D':player.x += speed;break;default:break;}}
}
#elif 1//2.使用Windows函数获取键盘输入//非阻塞函数,特别流畅//如果用字母,必须要用大写,大写可以识别大小写if (GetAsyncKeyState(VK_UP) || GetAsyncKeyState('W')){if(player.y > 0)player.y -= speed;}if (GetAsyncKeyState(VK_DOWN) || GetAsyncKeyState('S')){if (player.y < HIGHT-78)//78是飞机的高度player.y += speed;}if (GetAsyncKeyState(VK_LEFT) || GetAsyncKeyState('A')){if(player.x+59>0)// 119/2 = 59是控制飞机子弹在左右边界的时候可以打到敌机player.x -= speed;}if (GetAsyncKeyState(VK_RIGHT) || GetAsyncKeyState('D')){if (player.x-59 < WIDTH-119)//119是飞机的宽度player.x += speed;}
#endif // 0if (GetAsyncKeyState(VK_SPACE) && Timer(100,1))//{//创建一个子弹createBullet();}}
//产生敌机
void createenemy()
{for (int i = 0; i < ENEMY_NUM; i++){if (!enemy[i].live){enemy[i].live = true;enemy[i].x = rand()%(WIDTH-75);enemy[i].y = 0;enemyHP(i);printf("pos(%d,%d) %d %d\n", enemy[i].x, enemy[i].y, enemy[i].live, enemy[i].hp);break;}}
}//敌机的移动
void enemymove(int speed)
{for (int i = 0; i < ENEMY_NUM; i++){if (enemy[i].live){enemy[i].y += speed;if (enemy[i].y > HIGHT){enemy[i].live = false;}}}
}//开始打飞机
void playplane()
{for (int i = 0; i < ENEMY_NUM; i++){if (!enemy[i].live)continue;for (int k = 0; k < BULLET_NUM; k++){if (!bull[k].live)continue;if (bull[k].x > enemy[i].x && bull[k].x<enemy[i].x + enemy[i].width&& bull[k].y>enemy[i].y && bull[k].y < enemy[i].y + enemy[i].hight)//子弹在飞机的界面内//则代表子弹打住了飞机{bull[k].live = false;enemy[i].hp--;}}if (enemy[i].hp == 0){enemy[i].live = false;}}
}void showenemy()//测试飞机能不能一直往下面落
{for (int i = 0; i < ENEMY_NUM; i++){printf("pos(%d,%d) %d %d\n", enemy[i].x, enemy[i].y,enemy[i].live, enemy[i].hp);}
}int main()
{/*测试
circle(50, 50, 50);//画空心圆
setfillcolor(GREEN);//圆的颜色
fillcircle(100, 100, 50);///实心⚪*///创建图形窗口initgraph(WIDTH, HIGHT,SHOWCONSOLE);//宽长gameinit();//双缓冲绘图BeginBatchDraw();while (1){gameDraw();FlushBatchDraw();playermove(5);Bullmove(5);//防止飞机全部一起出来if (Timer(500,0)){//创建一个飞机createenemy();}if (Timer(20, 2)){enemymove(1);}playplane();}EndBatchDraw();return 0;
}