【C++/C 实现球球大作战】

这篇具有很好参考价值的文章主要介绍了【C++/C 实现球球大作战】。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

1.引言

这篇文章主要为大家详细介绍了C语言实现——《球球大作战项目》,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下。
【C++/C 实现球球大作战】,c++,c语言,开发语言
游戏介绍:
这是一个大球吃小球的世界,玩家的目标是要努力吃成最大的球球。在游戏一开始,玩家出现在地图上随机位置,地图里洒满了小彩豆,玩家吃掉小彩豆体积就会增大,当增大到比别人的球大时,就可以吃别人的球啦,当然也得躲避更大的球。小球速度快,大球速度慢。球球到达一定重量后,可以分身,一个球变成等大的两个球,可以再次分身,但是最多可以拥有16个分身。
效果图展示
【C++/C 实现球球大作战】,c++,c语言,开发语言

2.游戏设计:概述游戏的玩法和操作方式。

1.操作方式:
使用键盘控制球体的移动,如W、A、S、D 或者 上、下、左、右方向键。
2.玩法:
玩家控制的球体通过键盘控制进行移动,可以朝不同的方向移动以吞噬其他球体。
球体可以通过碰撞来吞噬其他较小的球体,从而增大自己的体积。

3.游戏实现

此处包含需要用到的头文件,一些声明和定义 声明一个结构体可以让食物,玩家和ai玩家用,避免重复的定义。

#include<stdio.h>
#include<easyx.h>//包含图形库头文件
#include<stdlib.h>
#include<time.h>
#include<math.h>
#define FOOD_NUM 200 //食物数量
#define WIDTH 1024
#define HEIGHT 640
#define HEIFGT  10
#define AI_NUM 10//人工智障数量
/*
做游戏:1.窗口,画图,贴图。。。。。easyx图形库的
食物:圆(圆心坐标,半径,颜色,是否存在的标志)
*/

并且要明确要定义的对象
如下

struct Ball//小球结构,玩家,食物,人工智障
{
	int x;
	int y;
	int r;//半径
	DWORD color;//颜色
	bool flag;//是否存在
};
struct Ball food[FOOD_NUM];
struct Ball player;//玩家
struct Ball ai[AI_NUM];

(1)函数 GameInit() 初始化游戏的函数。

1.设置随机数种子:通过调用 srand((unsigned)time(NULL)),将随机数种子设置为当前时间的秒数,以确保每次游戏运行时生成的随机数序列是不同的。

2.初始化食物数据:通过循环遍历每个食物球体,为每个食物设置随机的位置、半径、颜色,并将存在标志 flag 设置为 true,表示初始状态下食物存在。

3.初始化玩家数据:生成玩家球体的初始位置、半径、颜色,并将存在标志 flag 设置为 true。

4.初始化人工智能角色(AI)数据:通过循环遍历每个 AI 角色,为每个角色设置随机的位置、半径和颜色,并将存在标志 flag 设置为 true。

void GameInit()
{
	//设置随机数种子,时间是在不断变化的
	srand((unsigned)time(NULL));
	for (int i = 0;i < FOOD_NUM; ++i)
	{
		food[i].x = rand() % WIDTH;
		food[i].y = rand() % HEIGHT;
		food[i].flag = true;//刚开始食物存在
		food[i].r = rand() % 6 + 1;
		food[i].color = RGB(rand() % 256, rand() % 256, rand() % 256);
	}
	//初始化玩家数据
	player.x = rand() & WIDTH+1;
	player.y = rand() % HEIGHT+1;
	player.r = 18;//只要比食物大就可以
	player.flag = true;
	player.color= RGB(rand() % 256, rand() % 256, rand() % 256);
	//初始化ai
	for (int i = 0; i < AI_NUM; ++i)
	{
		ai[i].x = rand() % WIDTH;
		ai[i].y = rand() % HEIGHT;
		ai[i].r = rand() % 10 + 5;
		ai[i].flag = true;
		ai[i].color= RGB(rand() % 256, rand() % 256, rand() % 256);
	}
}

(2)函数 GameDraw() 用于绘制游戏场景的函数。

BeginBatchDraw() 函数开始绘制,这可以避免屏幕闪烁。

setbkcolor(WHITE) 设置背景色为白色。

cleardevice() 清空屏幕,将整个屏幕颜色填充为背景色。

循环遍历每个食物球体(FOOD_NUM 个),进行绘制。

如果 food[i].flag 为真,表示食物存在,则根据 food[i] 的坐标和半径,绘制一个填充圆,颜色为 food[i].color。
否则,表示食物已经消失,在随机位置重新生成一个食物。

void GameDraw()
{
	//防止闪屏
	BeginBatchDraw();
	setbkcolor(WHITE);
	cleardevice();//清屏
	for (int i = 0; i < FOOD_NUM; ++i)
	{
		if (food[i].flag)
		{
			setfillcolor(food[i].color);//设置填充颜色
			solidcircle(food[i].x, food[i].y,food[i].r);//画一个填充圆
		}
		else
		{
			food[i].x = rand() % WIDTH;
			food[i].y = rand() % HEIGHT;
			food[i].flag = true;//刚开始食物存在
			food[i].r = rand() % 6 + 1;
			food[i].color = RGB(rand() % 256, rand() % 256, rand() % 256);
		}
	}

绘制玩家球体:
设置填充颜色为 player.color。
根据玩家的坐标和半径,绘制一个填充圆。
设置文本颜色为绿色。
设置文本样式为大小为 30、字体为黑体。
设置文字背景为透明。
输出文字在玩家球体的位置上。

//绘制玩家
	setfillcolor(player.color);//设置填充颜色
	solidcircle(player.x, player.y, player.r);
	settextcolor(GREEN);
	settextstyle(30, 0, L"黑体");
	setbkmode(0);//设置文字背景透明
	outtextxy(player.x-50, player.y, L"我是最靓的崽仔");
	EndBatchDraw();
	//绘制ai
	for (int i = 0; i < AI_NUM;++i)
	{
		if (ai[i].flag)
		{
			setfillcolor(ai[i].color);
			solidcircle(ai[i].x, ai[i].y, ai[i].r);
		}
	}
}

(3)函数 keyControl(int speed) 负责处理键盘控制玩家球体移动

1.GetAsyncKeyState() 函数用于获取键盘消息,判断是否按下了特定的键。
2.键盘控制包括上、下、左、右四个方向键。根据按下的键,判断移动的方向并更新玩家球体的位置。
3.如果按下了 ‘W’ 键或者上方向键 (VK_UP),并且玩家球体的 y 坐标大于等于 0,则向上移动,即 player.y -= speed。
其他位置都如上
4.根据按键的状态和边界限制,函数根据移动速度 speed 更新玩家球体的 函数

void keyControl(int speed)
{
	//获取键盘消息,按的是哪一个键 
	if ((GetAsyncKeyState('W') || GetAsyncKeyState(VK_UP))&&player.y>=0)//按了上键
	{
		player.y -= speed;
	}
	if ((GetAsyncKeyState('S') || GetAsyncKeyState(VK_DOWN))&& player.y<=HEIGHT)//按了下键
	{  
		player.y += speed;
	}  
	if ((GetAsyncKeyState('A') || GetAsyncKeyState(VK_LEFT))&& player.x >= 0)//按了左键
	{  
		player.x -= speed;
	}  
	if ((GetAsyncKeyState('D') || GetAsyncKeyState(VK_RIGHT))&& player.x<=WIDTH)//按了右键
	{  
		player.x += speed;
	}
}

特别注意边界,不能越界

(4)函数 eatFood() 负责处理玩家吃食物

函数通过遍历所有的食物球体,检查玩家球体是否和食物重叠,并进行吃食物的操作。
对于每一个食物球体:
如果食物存在(food[i].flag 为真)且食物球体和玩家球体的距离小于等于玩家球体的半径 player.r,则表示玩家球体吃到了该食物。
吃到食物后,将食物的存在标志 food[i].flag 设置为假,表示食物消失。
将玩家球体的半径 player.r 增加吃到食物球体半径 food[i].r 的1/4,以增加玩家球体的大小。
其中为判断是否被吃,也就是距离问题,需要引入求距离函数

double distance(struct Ball a, struct Ball b)
{
	return sqrt((double)(a.x - b.x) * (a.x - b.x) + (a.y - b.y) * (a.y - b.y));
}

玩家吃食物

void eatFood()
{
	for (int i = 0; i < FOOD_NUM; ++i)
	{
		if (food[i].flag && distance(food[i], player) <= player.r)
		{
			food[i].flag = false;
			player.r += food[i].r / 4;
		}
	}
}

(5)函数 aiMove() 是用于AI角色移动

1.对于每个AI角色(AI_NUM 个):
2.如果AI角色存在(ai[i].flag 为真),则分别生成随机的X轴和Y轴偏移量。
3.X轴偏移量通过 rand() % 5 - 2 生成一个介于-2到2之间的随机数。
4.Y轴偏移量同样通过 rand() % 5 - 2 生成一个介于-2到2之间的随机数。
5.将随机生成的偏移量加到AI角色的当前位置上,实现随机移动的效果。

void aiMove()
{
	for (int i = 0; i < AI_NUM; ++i)
	{
		if (ai[i].flag)
		{
			ai[i].x += rand() % 5 - 2;  // 生成随机的X轴偏移量
			ai[i].y += rand() % 5 - 2;  // 生成随机的Y轴偏移量
		}
	}
}

以下是优化版本

void aiMove()
{
	for (int i = 0; i < AI_NUM; ++i)
	{
		if (ai[i].flag)
		{
			int xOffset = rand() % 5 - 2; // 生成随机的X轴偏移量
			int yOffset = rand() % 5 - 2; // 生成随机的Y轴偏移量

			ai[i].x += xOffset;
			ai[i].y += yOffset;

			// 限制AI角色移动的范围,确保不超出窗口边界
			if (ai[i].x < 0)
				ai[i].x = 0;
			else if (ai[i].x > WIDTH)
				ai[i].x = WIDTH;

			if (ai[i].y < 0)
				ai[i].y = 0;
			else if (ai[i].y > HEIGHT)
				ai[i].y = HEIGHT;
		}
	}
}

优化说明:
1.使用变量 xOffset 和 yOffset 存储随机生成的X轴和Y轴的偏移量,以减少重复的随机数生成。
2.限制AI角色移动的范围,确保AI角色的位置不会超出窗口的边界。通过比较偏移量后的位置和窗口宽度和高度的大小来限制移动范围。

(6)主函数 int main() 控制游戏的主循环和整体

int main()
{
	initgraph(WIDTH, HEIGHT);
	GameInit();
	while (1)
	{
		keyControl(1);
		GameDraw();
		eatFood();
		aiMove();
	}
	getchar();
	return 0;
}

4.最终代码

#include<stdio.h>
#include<easyx.h>//包含图形库头文件
#include<stdlib.h>
#include<time.h>
#include<math.h>
#define FOOD_NUM 200 //食物数量
#define WIDTH 1024
#define HEIGHT 640
#define HEIFGT  10
#define AI_NUM 10//人工智障数量
/*
做游戏:1.窗口,画图,贴图。。。。。easyx图形库的
食物:圆(圆心坐标,半径,颜色,是否存在的标志)
*/
struct Ball//小球结构,玩家,食物,人工智障
{
	int x;
	int y;
	int r;//半径
	DWORD color;//颜色
	bool flag;//是否存在
};
struct Ball food[FOOD_NUM];
struct Ball player;//玩家
struct Ball ai[AI_NUM];
void GameInit()
{
	//设置随机数种子,时间是在不断变化的
	srand((unsigned)time(NULL));
	for (int i = 0; i < FOOD_NUM; ++i)
	{
		food[i].x = rand() % WIDTH;
		food[i].y = rand() % HEIGHT;
		food[i].flag = true;//刚开始食物存在
		food[i].r = rand() % 6 + 1;
		food[i].color = RGB(rand() % 256, rand() % 256, rand() % 256);
	}
	//初始化玩家数据
	player.x = rand() & WIDTH + 1;
	player.y = rand() % HEIGHT + 1;
	player.r = 18;//只要比食物大就可以
	player.flag = true;
	player.color = RGB(rand() % 256, rand() % 256, rand() % 256);
	//初始化ai
	for (int i = 0; i < AI_NUM; ++i)
	{
		ai[i].x = rand() % WIDTH;
		ai[i].y = rand() % HEIGHT;
		ai[i].r = rand() % 10 + 5;
		ai[i].flag = true;
		ai[i].color = RGB(rand() % 256, rand() % 256, rand() % 256);
	}
}
//绘制
void GameDraw()
{
	//防止闪屏
	BeginBatchDraw();
	setbkcolor(WHITE);
	cleardevice();//清屏
	for (int i = 0; i < FOOD_NUM; ++i)
	{
		if (food[i].flag)
		{
			setfillcolor(food[i].color);//设置填充颜色
			solidcircle(food[i].x, food[i].y,food[i].r);//画一个填充圆
		}
		else
		{
			food[i].x = rand() % WIDTH;
			food[i].y = rand() % HEIGHT;
			food[i].flag = true;//刚开始食物存在
			food[i].r = rand() % 6 + 1;
			food[i].color = RGB(rand() % 256, rand() % 256, rand() % 256);
		}
	}
	//绘制玩家
	setfillcolor(player.color);//设置填充颜色
	solidcircle(player.x, player.y, player.r);
	settextcolor(GREEN);
	settextstyle(30, 0, L"黑体");
	setbkmode(0);//设置文字背景透明
	outtextxy(player.x-50, player.y, L"我是最靓的崽仔");
	EndBatchDraw();
	//绘制ai
	for (int i = 0; i < AI_NUM;++i)
	{
		if (ai[i].flag)
		{
			setfillcolor(ai[i].color);
			solidcircle(ai[i].x, ai[i].y, ai[i].r);
		}
	}
}
void keyControl(int speed)
{
	//获取键盘消息,按的是哪一个键 
	if ((GetAsyncKeyState('W') || GetAsyncKeyState(VK_UP))&&player.y>=0)//按了上键
	{
		player.y -= speed;
	}
	if ((GetAsyncKeyState('S') || GetAsyncKeyState(VK_DOWN))&& player.y<=HEIGHT)//按了下键
	{  
		player.y += speed;
	}  
	if ((GetAsyncKeyState('A') || GetAsyncKeyState(VK_LEFT))&& player.x >= 0)//按了左键
	{  
		player.x -= speed;
	}  
	if ((GetAsyncKeyState('D') || GetAsyncKeyState(VK_RIGHT))&& player.x<=WIDTH)//按了右键
	{  
		player.x += speed;
	}
}
//求两点之间的距离
double distance(struct Ball a, struct Ball b)
{
	return sqrt((double)(a.x - b.x) * (a.x - b.x) + (a.y - b.y) * (a.y - b.y));
}
//玩家吃食物
void eatFood()
{
	for (int i = 0; i < FOOD_NUM; ++i)
	{
		if (food[i].flag && distance(food[i], player) <= player.r)
		{
			food[i].flag = false;
			player.r += food[i].r / 4;
		}
	}
}
//ai移动
void aiMove()
{
	for (int i = 0; i < AI_NUM; ++i)
	{
		if (ai[i].flag)
		{
			ai[i].x += rand() % 5-2;
			ai[i].y += rand() % 5 - 2;
		}
	}
}
int main()
{
	initgraph(WIDTH, HEIGHT);
	GameInit();
	while (1)
	{
		keyControl(1);
		GameDraw();
		eatFood();
		aiMove();
	}
	getchar();
	return 0;
}

其他具体功能读者可自行添加。文章来源地址https://www.toymoban.com/news/detail-700756.html

到了这里,关于【C++/C 实现球球大作战】的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 【从零开始学Skynet】实战篇《球球大作战》(十二):场景代码设计(上)

            场景服务会处理绝大部分的游戏逻辑。新建 service/scene/init.lua ,开始编写相关代码。    场景中包含小球和食物这两种对象,先看看小球的实现。代码如下所示: 首先定义 balls 表 和 ball 类 , balls 表 会以 玩家 id 为索引,保存战场中各个小球的信息; 小球与玩

    2023年04月21日
    浏览(41)
  • 【从零开始学Skynet】实战篇《球球大作战》(十三):场景代码设计(下)

            《球球大作战》是一款服务端运算的游戏,一般会使用主循环程序结构,让服务端处理战斗逻辑。如下图所示,图中的 balls 和 foods 代表服务端的状态,在循环中执行 “ 食物生成 ”“ 位置更新 ” 和 “ 碰撞检 测” 等功能,从而改变服务端的状态。 scene 启动后

    2023年04月18日
    浏览(61)
  • 球球大作战 旋转合球脚本 - 用于蓝蝶模拟器(blueStacks5)

    键位截图 JS代码 下面是生成好的蓝蝶脚本代码,仅供参考

    2024年02月12日
    浏览(144)
  • 【开源游戏】Legends-Of-Heroes 基于ET 7.2的双端C#(.net7 + Unity3d)多人在线英雄联盟风格的球球大作战游戏。

    FlameskyDexive/Legends-Of-Heroes: A battle of balls game, lol style. 基于ET 7.2的双端C#(.net7 + Unity3d)多人在线英雄联盟风格的球球大作战。 (github.com)  一个LOL风格的球球大作战游戏,基于ET7.2,使用状态同步  基于C#双端框架[ET7.2],同步到ET主干详情请看日志。(https://github.com/egametang/ET) 注意:

    2024年02月03日
    浏览(59)
  • C语言调试大作战:与VS编译器共舞,上演一场“捉虫记”的艺术与科学

    少年们好,我是博主 那一脸阳光 ,我们接下来介绍C语言的调试和bug的分享。 引言: “如果你曾经在深夜与一串神秘莫测的C代码狭路相逢,彼此瞪大眼睛,犹如牛仔对决般紧张刺激;或者你曾试图驯服一段狂野不羁的循环,却发现自己陷入了一个深不见底的逻辑黑洞,那么

    2024年01月16日
    浏览(45)
  • 【C/C++小游戏】2048 大作战!(基于Easyx图形窗口实现)

    写在前面 游戏简介 Easyx 图形库 编写游戏 预编译代码 第一步:初始化棋盘 第二步:绘制棋盘 第三步:用户操作 第四步:封装函数 完整代码 效果展示 大家好! 本人是一个12岁六年级小学生,今年9月开始学习C++,曾经学过1年Python。 这是我的第一篇博客,决定分享一个游戏

    2024年02月10日
    浏览(45)
  • 球球大佬们帮忙分析win10蓝屏dmp!!

    背景:win10系统某天系统需要更新2H22还是哪个版本失败,后面玩大作的时候会突然蓝屏,例如2k23、荒野大镖客等,以前是不会的!现在日常使用和玩英雄联盟也不会蓝屏! 蓝屏代码:WHEA_UNCORRECTABLE_ERROR (124) 尝试解决:①新买了两条内存还是蓝屏;②拔掉内存条和显卡,橡皮

    2024年02月11日
    浏览(39)
  • 摸鱼大作战闯关

    查看源代码 查看源代码 F12 看源码   英语是真的烂 凯撒密码   十六进制 base64解密  创建post请求输入空密码 同上  

    2024年02月11日
    浏览(32)
  • Java贪吃蛇大作战

    作为Java新手小白,渴望学习一些好玩有趣的java程序 废话不多说,接下来我会一步一步实现java小程序:贪吃蛇大作战哦!  实现 Java贪吃蛇一共分四个步骤: 关于代码中的数值,因为小蛇的素材图片像素为25*25 ,游戏区域灰色方格为850*600         使用JFrame这个类,分别设

    2024年02月06日
    浏览(24)
  • 蛇蛇大作战

    AI蛇算法设计: 基本属性: 暂定6条AI蛇,互不攻击; 随机长度,颜色,位置;(rand函数) 移动: 方案一:BFS广度搜索最短路径找最近food,然后更新最近food位置; 方案二:rand随机位置+方向移动不碰壁and玩家player 吃食物: 与player玩家一致,吃后即刷新(保证10个数量food)

    2023年04月22日
    浏览(22)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包