UE4蓝图与C++交互——射击游戏中多武器系统的实现
回顾
学习UE4已有近2周的时间,跟着数天学院“UE4游戏开发”课程的学习,已经完成了UE4蓝图方面比较基础性的学习。通过UE4蓝图的开发,我实现了类似CS的单人版射击游戏,效果如下视频:
不得不说UE4蓝图功能的强大,无需写一句代码,就能实现一个基本的游戏玩法。并且使用门槛极低,只要熟悉蓝图的API,通过“拖拖,连连”就能完成游戏玩法的开发,对游戏策划(设计师)及其友好,与C++相比,生产效率极高。
多武器系统
目前的游戏设定是开场后,角色身上就自动装备了一把武器,为了实现类似于CS雪地地图一样的设定:开场没有武器,地图中摆放着多把武器,需要从地上拾取后才能使用,就需要将当前的武器系统进行扩展,创建多个蓝图来实现不同武器的模型、特效、属性等等。
武器玩法逻辑的处理
之前只有单个武器时,武器蓝图包含有玩法逻辑例如弹道计算、粒子显示、开枪动画、伤害控制等等,这些玩法(GamePlay)的逻辑是写在武器蓝图里面的。如果按照以前的设计,创建多个武器蓝图时,这些武器中的玩法逻辑也需要拷贝多份。
拷贝
然而,“拷贝”这种开发方式在程序编码中是非常不推荐的。所谓拷贝,意味着需要维护多份同样逻辑实现的代码,如果后续需要对该部分玩法进行调整 优化(策划又要改需求),那么拷贝了多少次,就需要修改多少次。例如,项目后期有50把武器(参考穿越火线),如果在前期为了省事将武器的玩法逻辑拷贝了49次,那么如果某天有该逻辑的修改需求时,就需要将该修改操作重复50次,这是让人崩溃的一件事。因此,在软件开发中,不要轻易使用Ctrl-C + Ctrl-V。
引用
那么该如何解决该问题呢?软件开发中常用的方法是将拷贝转变为引用。所谓引用,意思就是将公共的部分剥离出来,形成函数(方法),在需要该逻辑的地方引用该函数即可。如果后期有修改需求,只需要修改该函数一个地方,就可以实现多处逻辑被同时修改。同样,在UE4的蓝图中也提供了函数(Functions)这样的功能,通过创建函数可以将蓝图中公共的逻辑封装 ,从而实现多处引用。但是,UE4的蓝图是面向对象的,不同的武器(对象)之间是不能共用函数的,因此,将公共逻辑改为引用的方式是行不通的。
继承
既然蓝图是面向对象的,那么可以使用面向对象的编程特点:继承。所谓继承,就是将函数、变量封装到父类,从该父类集成出来的子类可以使用父类暴露出来的方法与属性。利用继承的特性,于是我们就可以从蓝图的Actor类中继承出Weapon类,而我们游戏中的各种武器,例如手枪、冲锋枪、狙击枪、火箭炮等等可以使用武器中封装的一些逻辑,比如开枪,换弹匣,等等,继承图如下:
可以看到,各种武器继承了Weapon之后,就拥有的子弹的变量,也拥有了开枪的函数,从而实现了武器逻辑的复用。
武器属性的设置
不同的武器除了模型(Mesh)不同以外,还有子弹数量、开枪动画、装弹动画、子弹撞击粒子、子弹伤害等等不同的属性,不同的武器这些属性都不同,而这些属性都需要在父类Weapon中进行处理。那么如何才能为不同的武器配置这些属性呢?这就涉及到C++变量如何暴露给蓝图使用。
根据官方文档:虚幻反射系统,C++中的变量可以被UPROPERTY()
宏修饰,就可以暴露给蓝图使用,还可以根据需要设定访问权限。
C++代码如下:
// 击中目标粒子
UPROPERTY(EditDefaultsOnly)
UParticleSystem* TargetFX;
// 开枪动画
UPROPERTY(EditDefaultsOnly)
UAnimationAsset* ShootAnim;
// 开枪间隔
UPROPERTY(EditDefaultsOnly)
float ShootInterval;
// 换弹匣动画
UPROPERTY(EditDefaultsOnly)
UAnimationAsset* ReloadAnim;
// 换弹匣时间
UPROPERTY(EditDefaultsOnly)
float ReloadTime;
// 每颗子弹伤害值
UPROPERTY(EditDefaultsOnly)
float Damage;
// 最大子弹数
UPROPERTY(EditDefaultsOnly)
int8 MaxBullet;
// 是否正在换弹匣
UPROPERTY(BlueprintReadOnly)
bool Reloading = false;
// 是否正在射击
UPROPERTY(BlueprintReadOnly)
bool Shooting = false;
// 当前子弹数(因为要暴露给蓝图获取,所以类型扩充到int32)
UPROPERTY(BlueprintReadOnly)
int32 CurrentBullet;
自动生成的蓝图设置如下:
可以看到,如果变量是粒子指针和动画的指针,蓝图中则直接生成了对应的可视化选择框,太方便了有木有。这不就是策划所需要的配置表吗,还是可视化的,再也不用担心把文件名配错了。
武器逻辑
除了变量,C++函数有类似的处理方法,宏UFUNCTION()
可以将函数暴露给蓝图,供蓝图调用。因此,上文中提到的换弹匣的逻辑,就可以移植到C++中,从而给予所有武器具有开枪与换弹匣能力。
开枪的核心代码如下(已精简):
// 获取坐标与朝向
World->GetFirstPlayerController()->GetPlayerViewPoint(Location, Rotation);
// 播放开枪动画
Mesh->PlayAnimation(ShootAnim, false);
// 计算终点坐标 = 起点坐标 + 方向 * 距离
FVector EndLocation = Location + Rotation.Vector() * 10000;
// 发射射线
World->LineTraceSingleByChannel(Result, Location, EndLocation, ECC_WorldStatic, ccq);
// 已击中
if (Result.Actor.IsValid())
{
// 播放粒子
FRotator EmitterRotation = FRotator(0, 0, 0);
AActor* Actor = Result.Actor.Get();
UGameplayStatics::SpawnEmitterAtLocation(Actor, TargetFX, Result.Location, EmitterRotation);
// 中弹的是Character
if (dynamic_cast<ACharacter*>(Actor) != NULL)
{
// 受伤害
ACharacter* Shooter = dynamic_cast<ACharacter*>(WeaponOwner);
UGameplayStatics::ApplyDamage(Actor, Damage, Shooter->GetController(), this, NULL);
}
}
接下来是换弹匣:
Reloading = true;
// 播放动画
Mesh->PlayAnimation(ReloadAnim, false);
// 设置延时回调
GetWorldTimerManager().SetTimer(ReloadTimer, this, &AWeapon::ReloadFinish, ReloadTime);
void AWeapon::ReloadFinish()
{
CurrentBullet = MaxBullet;
Reloading = false;
}
因为换弹匣并不是瞬间换好(按了R键需要等一定时间后子弹才会恢复),因此使用了定时器来实现。
遇到的问题
- 蓝图中绑定的Mesh无法传递到C++
我按照之前的方法,在C++中定义好骨骼Mesh指针:
// 武器mesh
UPROPERTY(EditDefaultsOnly)
USkeletalMeshComponent* MeshComponent;
编译之后,兴冲冲的跑到蓝图中准备选择Mesh,然而却发现蓝图中却没有出现MeshComponent这个字段,以为是C++代码没有编译到,于是就反复试了几次,结果还是没有。怎么办呢?怀疑是UE4的这个反射系统不支持USkeletalMeshComponent*
这种变量类型,把变量类型改为int32
后,果然,这个字段出现了。
SkeletalMeshComponent不行,那么父类MeshComponent呢?
// 武器mesh
UPROPERTY(EditDefaultsOnly)
UMeshComponent* MeshComponent;
然而,编译之后蓝图中还是没有......看来的确是不支持USkeletalMeshComponent*
或者UMeshComponent*
这种类型。怎么办呢?我突然想到,既然不支持在蓝图中直接选择默认值,那么在蓝图中调用set方法来设置该变量吧!
于是,我将C++代码修改为:
// 武器mesh
UPROPERTY(BlueprintReadWrite)
USkeletalMeshComponent* MeshComponent;
蓝图如下:
这样虽然实现了需求,但是需要在每一把武器蓝图中做一样的调用,如果有50把武器呢?100把武器呢?这样做明显与我的期望不一致,因此这种方法虽然可行,但是不可取。
还有其他办法吗?通过查询UE4 C++的API,我找到了AActor::GetComponentByClass
这个函数,官方文档的描述是:
Searches components array and returns first encountered component of the specified class
即查找并获取本Actor的指定类型的组件的第一个。这不正是我需要的吗?如果我指定查找类型为USkeletalMeshComponent,而每个武器只有一个USkeletalMeshComponent,那这样不就从蓝图中获取到了绑定的Mesh吗?实现代码如下:
void AWeapon::BeginPlay()
{
Super::BeginPlay();
// 获得Mesh
MeshComponent = dynamic_cast<USkeletalMeshComponent*>(GetComponentByClass(USkeletalMeshComponent::StaticClass()));
}
通过这种方式,实现了武器的全部逻辑从蓝图中去除。当新加一件武器时,只需要在武器蓝图属性中配置动画、粒子、子弹容量等属性后,就可以直接在游戏中使用。
蓝图与C++选择的思考
既然UE4同时支持蓝图与C++,那么我们在开发时应该如何选择呢?官方文档有如下的解释:
程序员利用C++即可添加基础Gameplay系统,然后设计师可基于这些系统进行构建或利用这些系统为某个特定关卡或游戏本身创建自定义Gameplay。
也就是说,程序员用C++开发一些基础的系统,例如本文当中的武器系统(Weapon),设计师(策划)即可利用该武器系统在蓝图上进行武器的扩充,设计出不同的武器;设计师(策划)也可以利用C++开发的基础系统,将这些系统在蓝图上进行组装,以构建更丰富的玩法系统。除此之外,
- 蓝图的可重构性非常非常非常低,如果某个模块的逻辑比较复杂,就会出现各种线条乱飞,非常的凌乱,过段时间再过来看,可能作者自己都看不明白了。因此我建议,对于比较复杂的逻辑,最好使用C++来实现,如果一定要用蓝图,请将该部分逻辑拆分为几段小逻辑来实现(充分利用蓝图的函数与宏)。
- 蓝图本身是作为二进制文件来保存(.uasset),在版本管理工具中(Git)无法进行差异性对比,如果对于某段频繁修改(升级)的逻辑,又想看到每次修改的变化,最好也使用C++来实现,可以对比每个版本的修改内容。
- 对于复杂的数学运算或循环次数较多的逻辑,也最好采用C++来实现,以保证运行效率。
UE4蓝图与C++交互——射击游戏中多武器系统的实现的更多相关文章
- Unity优化方向——优化Unity游戏中的脚本(译)
原文地址:https://unity3d.com/cn/learn/tutorials/topics/performance-optimization/optimizing-scripts-unity ...
- 少儿编程Scratch第四讲:射击游戏的制作,克隆的奥秘
上周的宇宙大战射击游戏中,我们只完成了宇宙飞船发射子弹的部分.还未制作敌对方.这周制作了敌方-飞龙,飞龙随机在屏幕上方出现,如果被子弹打中,则得分,飞龙和子弹都消失. 敌方:飞龙:计分. 目的 目的: ...
- 来玩一局CS吗?UE4射击游戏的独立服务器构建
前言 根据UE4官方文档的介绍,UE4引擎在架构时就已经考虑到了多人游戏的情景,多人游戏基于客户端-服务器模式(CS模式).也就是说,会有一个服务器担当游戏状态的主控者,而连接的客户端将保持近似的 ...
- 游戏中的沉浸(Flow in Games)
转自:https://www.jianshu.com/p/4c52067f6594 作者:陈星汉(JenovaChen) 本论文的主旨在于提供一种独特的方法论,用以指导游戏设计中的以玩家为中心的动态难 ...
- 使用Cocos2dx-JS开发一个飞行射击游戏
一.前言 笔者闲来无事,某天github闲逛,看到了游戏引擎的专题,引起了自己的兴趣,于是就自己捣腾了一下Cocos2dx-JS.由于是学习,所谓纸上得来终觉浅,只是看文档看sample看demo,并 ...
- 喵的Unity游戏开发之路 - 推球:游戏中的物理
很多童鞋没有系统的Unity3D游戏开发基础,也不知道从何开始学.为此我们精选了一套国外优秀的Unity3D游戏开发教程,翻译整理后放送给大家,教您从零开始一步一步掌握Unity3D游戏开发. 本文不 ...
- UE4蓝图AI角色制作(四)之Gameplay调试器
8. 寻路网格体和Gameplay调试器 为了及时识别出AI系统中的导航问题,UE4提供了一个工具用来解决这类问题,它叫Gameplay调试器.打开项目设置,在左侧找到"引擎",然 ...
- UE4蓝图编程的第一步
认识UE4蓝图中颜色与变量类型: UE4中各个颜色对应着不同的变量,连接点和连线的颜色都在表示此处是什么类型的变量.对于初学者来说一开始看到那么多连接点, 可能会很茫然,搞不清还怎么连,如果知道了颜色 ...
- 用canvas制作酷炫射击游戏--part3
今天介绍下 游戏中的sprite模块,也就是构建玩家及怪物的模块.有了这个模块,就可以在咱们的游戏里加入人物了. 想必用过css的朋友都知道sprite,一种将需要加载的图片拼接在一张图里以减少请求的 ...
随机推荐
- unzip 命令指定解压路径
在使用unzip进行文件包解压,可以用来解压zip/jar/war包类型,有时解压时需要解压到指定路径时可以使用参数 -d 来指定,例如: unzip services-bak.jar -d ./we ...
- html实现打印预览效果
前面说到利用lodop插件进行打印设置,那个应用于打印快递面单,或者跟快递面单相似场景的情况. 今天的利用html快速打印出A4纸大小的场景,例如:合同.静态文本等. 效果如下: 方式一 1.设置di ...
- xampp修改mysql 启动脚本
打开xmapp,点击mysql对应的config按钮进入my.ini文件,如图所示: 修改mysqld服务的port参数3306为你想要设置的port,如图2所示: 重新启动mysql服务即可用客户端 ...
- Activiti(1) - TaskRuntime API 入门
目录 TaskRuntime API pom.xml 注册TaskRuntime实例 角色与分组 任务事件监听器 DemoApplication 源码 Activiti 是一个自动化工作流框架.它能帮 ...
- WCF尝试创建与发布IIS(含问题描述)
技术贴技术贴就直接讲技术来,客套的话我也不多说了,各位看官包涵包涵. 跟着园内高手一步一步发布成功,欣喜若狂之际,发个贴纪念纪念一下. 废话不多说,不正确之处,还望大家积极指出,共同进步.哈哈~~~ ...
- PHP 微信机器人 Vbot 结合 Laravel 基础入门及实例
新来了项目,需要能监听指定微信群的消息并进行转发.基于 PHP 7 的 web 微信机器人 Vbot 可以满足需求.Vbot 本质上就是实现了登录网页版微信来进行自动回复.群管理等等操作. githu ...
- js模拟下拉菜单-键盘、鼠标(代码详解)
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- 深圳市网络安全中心发出通告,TeamViewer已被APT41黑客攻破
上期我们讲了东南亚赌局为什么都是福建老板了,这次来介绍下黑客组织APT41 ,这个组织在 HT界 比较出名,很早之前是匿名在地下交易所的,而在近年频繁出现在大众视野中,这不,刚刚又把我们常用的远程工具 ...
- 一个基于protobuf的极简RPC
前言 RPC采用客户机/服务器模式实现两个进程之间的相互通信,socket是RPC经常采用的通信手段之一.当然,除了socket,RPC还有其他的通信方法:http.管道...网络开源的RPC框架也比 ...
- 关于jQery中$.Callbacks()的理解
$.Callbacks()主要使用了回调,而说到回调又不得不说javascript的事件循环机制了. 所以想了解回调最好先看看js运行机制. $.Callbacks()可以理解为创建一个回调队列 va ...