C/C++实现球球大作战(高仿版),全源码分享带你轻松完成

Posted 一起学编程

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了C/C++实现球球大作战(高仿版),全源码分享带你轻松完成相关的知识,希望对你有一定的参考价值。

每天一个C语言小项目,提升你的编程能力! 

 《球球大作战》虽然在玩法上类似于大球吃小球的模式看起来很单薄。但是在游戏过程中会出现无数种意外情况,这就需要玩家运用一系列策略来达到不被吃和吃掉别人球的目的,大大增加了游戏的耐玩性。

一个人的话想要实现复刻球球太困难了,所以这是仿照成熟版球球大作战写的简易版小游戏,有食物、敌人,甚至像和平精英一样加了一层外面的毒圈。

游戏操作起来很简单,用 A S D W 四个键控制球的移动方向。

地图大小是屏幕的16倍,吃完所有敌人就胜利。记住不要被敌人吃掉哦!

效果图展示:

完整的游戏源代码如下:

#include <graphics.h>
#include <conio.h>
#include <time.h>
#include <math.h>
#include <wchar.h>

#define WIDTH		1024			// 屏幕宽
#define HEIGHT		576				// 屏幕高
#define MAPW		(WIDTH * 4)		// 地图宽
#define MAPH		(HEIGHT * 4)	// 地图高
#define AINUM       100				// AI 数量
#define FNUM        2000			// FOOD 数量
#define PTIME       180				// 毒圈刷新时间
#define DISTANCE(x1, y1, x2, y2)	(sqrt((x1 - x2) * (x1 - x2) + (y1 - y2) * (y1 - y2)))

struct FOOD

	bool		eat;
	COLORREF	color;		// 颜色
	int			x, y;		// 坐标
	char		type;
;

struct BALL					// 定义小球结构

	bool		life;		// 生命
	COLORREF	color;		// 颜色
	int			x, y;		// 坐标
	float		r;			// 半径
;

FOOD food[FNUM];								// 食物
BALL mover =  1, RGB(0, 0, 0), 0, 0, 0 ;	// 玩家
BALL ai[AINUM] =  1, RGB(0, 0, 0), 0, 0, 0 ;	// 机器大军

void move(BALL* ball);				// 玩家移动
void draw();						// 绘图
void start();						// 游戏开始
void setall();						// 初始化数据
void AI();							// AI
void Food();						// 食物
void delay(DWORD ms);				// 绝对延时

DWORD* pBuffer;						// 显示缓冲区指针
int eaten = 0;						// 吃 AI 的数量
int ai_eaten = 0;					// AI 吃 AI的数量
int lx = -20, ly = MAPH + 20, rx = MAPW + 20, ry = -20;		// 毒的位置
int relx = -20, rely = MAPH + 20, rerx = MAPW + 20, rery = -20;		// 毒的位置
float asp = 1;						// 缩放因子
float Time = 0;						// 时间

int main()

	initgraph(WIDTH, HEIGHT);
	start();
	setall();				// 游戏初始化

	BeginBatchDraw();
	while (true)
	
		move(&mover);		// 玩家移动
		AI();
		Food();
		draw();				// 绘图
		FlushBatchDraw();	// 显示缓存的绘制内容
		delay(20);			// 绝对延迟
	


void move(BALL* ball)

	if (ball->r <= 0)	ball->life = false;

	if (ball->life == false)			// 判定游戏是否接束
	
		HWND hwnd = GetHWnd();
		MessageBox(hwnd, _T("你被吃了"), _T("游戏结束"), MB_OK | MB_ICONEXCLAMATION);	// 结束
		closegraph();
		exit(0);
	

	if (eaten + ai_eaten == AINUM)	// 是否吃掉所以 AI
	
		HWND hwnd = GetHWnd();
		MessageBox(hwnd, _T("恭喜过关"), _T("游戏结束"), MB_OK | MB_ICONEXCLAMATION);	// 结束
		closegraph();
		exit(0);
	

	if (ball->x > (MAPW - ball->r) || ball->x - ball->r < 0 || ball->y - ball->r < 0 || ball->y >(MAPH - ball->r))
		ball->r -= 0.1f;

	for (int i = 0; i < AINUM; i++)	// 玩家吃 AI 判定
	
		if (ball->r >= ai[i].r)
		
			if (ai[i].life == 0) continue;

			if (DISTANCE(ball->x, ball->y, ai[i].x, ai[i].y) < (4 / 5.0 * (ball->r + ai[i].r)))
			
				ai[i].life = 0;		// 吃掉
				ball->r += (ai[i].r * ai[i].r / 2) / ball->r;  // 推理过程: pai * ball->r*ball->r += pai*ai[i].r*ai[i].r;   πr*r=πr*r+πai[i].r*ai[i].r
				eaten++;
			
		
	

	for (int n = 0; n < FNUM; n++)			// 玩家吃食物
	
		if (food[n].eat == 0) continue;

		if (DISTANCE(ball->x, ball->y, food[n].x, food[n].y) < ball->r)
		
			ball->r += 4 / ball->r;			// 增加面积
			food[n].eat = 0;				// 食物被吃
		
	

	static int mx = 0, my = 0;			// 记录偏移量
	if (GetAsyncKeyState(65) & 0x8000)  ball->x -= 2;	mx += 2; 	//左边
	if (GetAsyncKeyState(87) & 0x8000)  ball->y -= 2;	my += 2; 	//上面
	if (GetAsyncKeyState(83) & 0x8000)  ball->y += 2;	my -= 2; 	//下面
	if (GetAsyncKeyState(68) & 0x8000)  ball->x += 2;	mx -= 2; 	//右边
	setorigin(mx, my);					// 坐标修正


void AI()

	for (int i = 0; i < AINUM; i++)
	
		if (ai[i].r > mover.r)			// AI 吃玩家
		
			if (DISTANCE(mover.x, mover.y, ai[i].x, ai[i].y) < 2 / 3.0 * ai[i].r + mover.r)
			
				ai[i].r += (mover.r * mover.r) / ai[i].r;
				mover.life = 0;
				mover.r = 0;
			
		

		for (int j = 0; j < AINUM; j++)	// AI 吃 AI
		
			if (ai[i].r > ai[j].r)
			
				if (ai[j].life == 0) continue;

				if (DISTANCE(ai[i].x, ai[i].y, ai[j].x, ai[j].y) < 4 / 5.0 * ai[i].r + ai[j].r)
				
					ai[i].r += (ai[j].r * ai[j].r) / ai[i].r;		// 面积增加
					ai[j].life = 0;
					ai[j].r = 0;
					ai_eaten++;
				
			
		

		double min_DISTANCE = 100000;
		int min = -1;
		for (int k = 0; k < AINUM; k++)		// AI 靠近 AI
		
			if (ai[i].r > ai[k].r && ai[k].life == 1)
			
				if (DISTANCE(ai[i].x, ai[k].x, ai[i].y, ai[k].y) < min_DISTANCE)
				
					min_DISTANCE = DISTANCE(ai[i].x, ai[k].x, ai[i].y, ai[k].y);
					min = k;
				
			
		

		if ((min != -1) && (rand() % 2 == 1))
		
			if (rand() % 2)
			
				if (ai[i].x < ai[min].x) ai[i].x++;
				else ai[i].x--;
			
			else
			
				if (ai[i].y < ai[min].y) ai[i].y++;
				else ai[i].y--;
			
		

		for (int n = 0; n < FNUM; n++)		// AI 吃食物
		
			if (food[n].eat == 0) continue;

			if (DISTANCE(ai[i].x, ai[i].y, food[n].x, food[n].y) < ai[i].r)
			
				ai[i].r += 4 / ai[i].r;
				food[n].eat = 0;
			
		
	


void Food()

	for (int i = 0; i < FNUM; i++)			// 食物重新生成
	
		if (food[i].eat == 0)
		
			food[i].eat = 1;
			food[i].color = RGB(rand() % 256, rand() % 256, rand() % 256);
			food[i].x = rand() % MAPW;
			food[i].y = rand() % MAPH;
			food[i].type = rand() % 10 + 1;
		
	


void draw()

	clearcliprgn();
	setlinestyle(PS_SOLID | PS_JOIN_BEVEL, 20);	// 改笔的颜色、状态
	setlinecolor(RGB(0, 100, 0));
	line(relx, rely, relx, rery);	// 左竖
	line(relx, rely, rerx, rely);	// 上横
	line(relx, rery, rerx, rery);	// 下横
	line(rerx, rery, rerx, rely);	// 右竖
	setfillcolor(GREEN);

	if (mover.x - 0.5 * WIDTH / asp < relx)		floodfill(relx - 11, mover.y, RGB(0, 100, 0));
	if (mover.x + 0.5 * WIDTH / asp > rerx)		floodfill(rerx + 11, mover.y, RGB(0, 100, 0));
	if (mover.y - 0.5 * HEIGHT / asp < rery)		floodfill(mover.x, rery - 11, RGB(0, 100, 0));
	if (mover.y + 0.5 * HEIGHT / asp > rely)		floodfill(mover.x, rely + 11, RGB(0, 100, 0));

	setlinecolor(WHITE);			// 改笔颜色   状态
	setlinestyle(PS_NULL);

	for (int i = 0; i < FNUM; i++)	// 画出食物
	
		if (food[i].eat == 0) continue;
		setfillcolor(food[i].color);
		switch (food[i].type)		// 形状
		
		case 1:		solidellipse(food[i].x, food[i].y, food[i].x + 2, food[i].y + 4);				break;
		case 2:		solidellipse(food[i].x, food[i].y, food[i].x + 4, food[i].y + 2);				break;
		case 3:		solidrectangle(food[i].x, food[i].y, food[i].x + 4, food[i].y + 2);				break;
		case 4:		solidrectangle(food[i].x, food[i].y, food[i].x + 2, food[i].y + 4);				break;
		case 5:		solidroundrect(food[i].x, food[i].y, food[i].x + 2, food[i].y + 4, 2, 2);		break;
		case 6:		solidroundrect(food[i].x, food[i].y, food[i].x + 4, food[i].y + 2, 2, 2);		break;
		case 7:		solidroundrect(food[i].x, food[i].y, food[i].x + 4, food[i].y + 2, 4, 2);		break;
		case 8:		solidroundrect(food[i].x, food[i].y, food[i].x + 4, food[i].y + 2, 2, 4);		break;
		case 9:		solidroundrect(food[i].x, food[i].y, food[i].x + 4, food[i].y + 2, 1, 1);		break;
		case 10:	fillcircle(food[i].x, food[i].y, 4);										break;
		
	

	for (int i = 0; i < AINUM; i++)	// 画 AI
	
		if (ai[i].life == 0) continue;
		setfillcolor(ai[i].color);
		fillcircle(ai[i].x, ai[i].y, int(ai[i].r + 0.5));
	

	setfillcolor(mover.color);		// 画玩家
	fillcircle(mover.x, mover.y, int(mover.r + 0.5));

	IMAGE map(150, 100);			// 小地图
	SetWorkingImage(&map);
	setbkcolor(RGB(120, 165, 209));	// 浅灰色背景
	cleardevice();

	for (int i = 0; i < AINUM; i++)	// 画 AI
	
		if (ai[i].life == 0) continue;
		setfillcolor(ai[i].color);
		fillcircle(ai[i].x * 150 / WIDTH / 4, ai[i].y * 100 / HEIGHT / 4, int(ai[i].r / 28 + 0.5));
	

	setfillcolor(mover.color);		// 画玩家
	fillcircle(mover.x * 150 / WIDTH / 4, mover.y * 100 / HEIGHT / 4, int(mover.r / 28 + 0.5));
	setlinecolor(RGB(0, 100, 0));

	if (lx != rx && ly != rx)
	
		line(lx * 150 / WIDTH, ly * 100 / HEIGHT, lx * 150 / WIDTH, ry * 100 / HEIGHT);	// 左竖
		line(lx * 150 / WIDTH, ly * 100 / HEIGHT, rx * 150 / WIDTH, ly * 100 / HEIGHT);	// 上横
		line(lx * 150 / WIDTH, ry * 100 / HEIGHT, rx * 150 / WIDTH, ry * 100 / HEIGHT);	// 下横
		line(rx * 150 / WIDTH, ry * 100 / HEIGHT, rx * 150 / WIDTH, ly * 100 / HEIGHT);	// 右竖
	

	setfillcolor(GREEN);
	floodfill(lx - 11, ly - 11, RGB(0, 100, 0));

	SetWorkingImage();			// 恢复绘图背景
	putimage(mover.x + int(0.5 * WIDTH) - 150, mover.y - int(0.5 * HEIGHT), 150, 100, &map, 0, 0);						// 画出小地图
	setlinecolor(LIGHTBLUE);	// 改笔颜色   状态
	setlinestyle(PS_SOLID | PS_JOIN_BEVEL, 4);
	line(mover.x + int(0.5 * WIDTH) - 151, mover.y - int(0.5 * HEIGHT), mover.x + int(0.5 * WIDTH) - 151, mover.y - int(0.5 * HEIGHT) + 99);	// 地图边框线
	line(mover.x + int(0.5 * WIDTH) - 151, mover.y - int(0.5 * HEIGHT) + 99, mover.x + int(0.5 * WIDTH), mover.y - int(0.5 * HEIGHT) + 99);	// 地图边框线

	setlinestyle(PS_NULL);		// 恢复笔
	wchar_t str[32];
	swprintf_s(str, L"质量:%.1f  击杀:%d", mover.r, eaten);
	settextcolor(BLUE);			// 改字体
	outtextxy(mover.x - int(0.5 * WIDTH), mover.y - int(0.5 * HEIGHT), str);
	settextcolor(BLUE);			// 改字体
	outtextxy(mover.x - 36, mover.y - 8, _T("作者:无言"));


void setall()

	srand((unsigned)time(NULL));		// 随机数
	mover.color = RGB(rand() % 256, rand() % 256, rand() % 256);	// 随机颜色
	mover.life = 1;					// 统统赋初值
	mover.x = int(WIDTH * 0.5);
	mover.y = int(HEIGHT * 0.5);
	mover.r = 20;

	for (int i = 0; i < AINUM; i++)		// AI 的
	
		ai[i].life = 1;
		ai[i].color = RGB(rand() % 256, rand() % 256, rand() % 256);
		ai[i].r = float(rand() % 10 + 10);
		ai[i].x = rand() % (MAPW - int(ai[i].r + 0.5)) + int(ai[i].r + 0.5);
		ai[i].y = rand() % (MAPH - int(ai[i].r + 0.5)) + int(ai[i].r + 0.5);
	

	for (int i = 0; i < FNUM; i++)		// 食物的
	
		food[i].eat = 1;
		food[i].color = RGB(rand() % 256, rand() % 256, rand() % 256);
		food[i].x = rand() % MAPW;
		food[i].y = rand() % MAPH;
		food[i].type = rand() % 10 + 1;
	

	pBuffer = GetImageBuffer(NULL);	// 获取显示缓冲区指针
	setbkcolor(WHITE);				// 白色背景
	cleardevice();					// 初始化背景
	settextcolor(LIGHTRED);			// 改字体
	setbkmode(TRANSPARENT);
	settextstyle(16, 0, _T("宋体"));


void delay(DWORD ms)				// 绝对延时

	static DWORD oldtime = GetTickCount();

	while (GetTickCount() - oldtime < ms)
		Sleep(1);

	oldtime = GetTickCount();


void start()

	setbkcolor(WHITE);		// 白色背景
	cleardevice();			// 初始化背景
	settextcolor(RED);		// 改字体
	setbkmode(TRANSPARENT);
	settextstyle(128, 0, _T("宋体"));
	outtextxy(40, 20, _T("仿制球球大作战"));
	settextstyle(32, 0, _T("宋体"));
	outtextxy(740, 135, _T("Ver 1.6"));
	settextcolor(BLUE);		// 改字体
	outtextxy(304, 240, _T("W上移 S下移 A左移 D右移"));
	outtextxy(112, 340, _T("躲避大球   追补小球   贪吃食物   增强实力"));
	settextcolor(BLACK);	//改字体
	settextstyle(32, 0, _T("宋体"));
	outtextxy(384, 500, _T("按任意键开始游戏"));
	settextstyle(20, 0, _T("宋体"));
	outtextxy(810, 10, _T("作者粉丝群: 734106058"));
	_getch();

大家赶紧去动手试试吧!

此外,我也给大家分享我收集的其他资源,从最零基础开始的教程到C语言C++项目案例,帮助大家在学习C语言的道路上披荆斩棘!

 

 整理分享(多年学习的源码、项目实战视频、项目笔记,基础入门教程)最重要的是你可以在群里面交流提问编程问题哦!

欢迎转行和学习编程的伙伴,利用更多的资料学习成长比自己琢磨更快哦!

(资料领取 ↓↓↓↓↓↓)

以上是关于C/C++实现球球大作战(高仿版),全源码分享带你轻松完成的主要内容,如果未能解决你的问题,请参考以下文章

C语言项目实战:《球球大作战》零基础项目 + 源码示例

C语言高仿贪吃蛇大作战,800行代码就能实现,结尾有源码~

C语言高仿贪吃蛇大作战,800行代码就能实现,结尾有源码~

球球大作战代码怎么写

球球大作战符号大全

球球大作战 名字白色代码是多少