用VS开发一款“飞机大战“单机游戏<C++>

显示界面如上图所示

自己找的背景和飞机素材,先将素材奉上.

 

接下来我先简单分析一下这个单机游戏的运行逻辑:

就像显示界面所显示的那样,我们想要实现的是自己的飞机在发射子弹(子弹在上图没显示),然后当子弹射到敌方飞机,这里设置了两种类型的飞机,如果读者想定义更多类型的,直接添加属性就可以.当子弹射到飞机的时候,飞机的血量减少,就像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是飞机在窗口中的坐标
			else
			putimage(enemy[i].x, enemy[i].y, &img_enemy[1], SRCINVERT);
		}
	}

}

绘制背景,绘制自己飞机,绘制子弹,绘制敌方飞机都采用putimage()函数就可以了

需要注意的有可能就是对于敌方飞机和子弹只有live是true的时候才绘制,而对于敌方飞机还区分了两种类型的飞机,加一个if-else输出对应的敌方飞机就可以了;

第五步,第四步我们绘制的图片是个啥,是初始化的位置,是一个静态的图,我们要实现的动态图,那接下来我们应该干嘛????????

当然是让子弹,自己飞机和敌方飞机都动起来了,也就是创建动画

首先让自己飞机动起来

void playermove(int speed)
{
#if 0

	if (_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 // 0
	
	if (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是飞机在窗口中的坐标
			else
			putimage(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 0

	if (_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 // 0
	
	if (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;
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值