44. UE5 RPG 初始化敌人的属性

在正常的游戏中,我们应该考虑如何去初始化角色属性,并且要给角色分好类型。比如,在我们游戏中,我们如何去初始化小兵的属性,并且还要实现小兵随着等级的增长而增加属性。而且就是小兵也有类型的区分,比如我们将在后面设置小兵分为三种:战士,游侠,法师。而小兵的属性实现,一般需要策划进行配表实现,将小兵的属性统一性的存入到表格导入项目。
我们将使用枚举去定义敌人的类型并创建一个ECharacterClass,并创建一个DataAsset命名为UCharacterClassInfo用来设置每种类型使用的属性初始化的GE。对于随着等级变动的基础属性,我们将使用曲线表格去生成所有等级的数据。并且,在数据中,我们去设置敌人需要附带的技能。
所以,接下来,我们实现的步骤是:

  1. 创建一个枚举ECharacterClass来定义角色的种类
  2. 创建一个DataAsset类 UCharacterClassInfo用来设置角色所使用得数据
  3. 创建曲线表格,用来定义角色属性跟随等级变化的值
  4. 创建设置角色属性的GE
  5. 通过函数实现角色应用设置的数据。

创建DataAsset

首先,我们创建角色使用的DataAsset,打开UE创建一个新类,继承DataAsset,并将其命名为CharacterClassInfo
在这里插入图片描述
在项目中,首先创建我们第一步需要的枚举,这里创建的三个职业的枚举

//角色职业类型的枚举
UENUM(BlueprintType)
enum class ECharacterClass : uint8
{
	Elementalist, //法师
	Warrior, //战士
	Ranger //游侠
};

接着创建一个结构体,用于根据不同的职业选择不同的结构体,这是只添加了第一项,就是基础的属性设置,后面我们会增加更多的内容。

USTRUCT()
struct FCharacterClassDefaultInfo
{
	GENERATED_BODY()

	UPROPERTY(EditDefaultsOnly, Category="Class Defaults")
	TSubclassOf<UGameplayEffect> PrimaryAttributes;
};

有了枚举和角色初始化的结构体以后,我们就可以创建DataAsset类了,在数据类里,我们首先创建一个Map类型,用于创建不同职业对应的基础属性GE,然后对于SecondaryAttributes和VitalAttributes,我们共用一套。最后增加一个函数,用于通过职业类型去获取基础的属性GE。

/**
 * 根据职业选择初始化角色的数据
 */
UCLASS()
class AURA_API UCharacterClassInfo : public UDataAsset
{
	GENERATED_BODY()

public:

	UPROPERTY(EditDefaultsOnly, Category="Class Defaults")
	TMap<ECharacterClass, FCharacterClassDefaultInfo> CharacterClassInformation;
	
	UPROPERTY(EditDefaultsOnly, Category="Common Class Defaults")
	TSubclassOf<UGameplayEffect> SecondaryAttributes;
	
	UPROPERTY(EditDefaultsOnly, Category="Common Class Defaults")
	TSubclassOf<UGameplayEffect> VitalAttributes;

	//通过枚举获取对应的初始化类
	FCharacterClassDefaultInfo GetClassDefaultInfo(ECharacterClass CharacterClass);
};

函数的视线,就是通过Map返回对应的类型的GE。

FCharacterClassDefaultInfo UCharacterClassInfo::GetClassDefaultInfo(ECharacterClass CharacterClass)
{
	return CharacterClassInformation.FindChecked(CharacterClass);
}

创建完成后,我们编译打开UE,在里面创建一个可配置的DataAsset
在这里插入图片描述
选择我们前面创建的类
在这里插入图片描述
打开以后,就是需要我们填充的值,这里有个细节,就是如果你得枚举第一项就选择枚举默认值,那么后面将无法添加,因为Map类型的Key不允许相同,所以你可以先增加后面的枚举类型,然后再最后设置默认枚举值。
在这里插入图片描述

创建对应的GE

创建了数据以后,我们需要创建对应的GE文件去填充,但是GE里面的数据需要根据等级去变动,所以,我们需要通过数据表格去实现这个功能。
首先我们创建对应的GE,之前,我们创建了一套主角使用的GE,主要属性是设置的固定值,虽然后续我们不会这么使用,刚好,直接使用它复制三个对应三个职业使用,然后创建数据图表,让GE根据等级去数据图表中获取数据。
在这里插入图片描述
然后是它的SecondaryAttributes和VitalAttribues,我们在主角上使用的次级属性设置的GE是Infinite类型的,复制一个修改成Instant类型的。因为敌人在我们的游戏设计中,它们是不会升级也不会变动的,所以不需要实时监听数值变化。而VitalAttributes则直接共用一套就行,毕竟就是最后在生成将血量和蓝量填充到最满,不需要额外制作。
在这里插入图片描述

创建GE使用的数据表格

为了能实现敌人的初始属性,我们将创建曲线表格来实现角色等级增长带来的属性数值的变化。我们在实现时,只要给敌人设置不同的等级,它就会生成不同的属性的敌人。
在这里,我们选择使用曲线表格,曲线表格的好处是可以自动填充中间的数据,并且可以过渡的更好。
在这里插入图片描述
创建曲线表格时,它会提示我们创建什么类型的曲线,
Constant就是没有过渡,到了位置显示固定值,一般不会使用
Linear就是线性过渡
Cubic就是贝塞尔曲线,它会圆滑的过渡,这里我们选择Cubic
在这里插入图片描述
曲线表格推荐使用CT开头,比如我们创建的法师的类型的就叫做CT_PrimaryAttributes_Elementalist
在这里插入图片描述
打开曲线表格,我们将四个属性都添加进去,然后选择曲线编辑
在这里插入图片描述

选中一项属性,然后鼠标右键添加关键帧
在这里插入图片描述
可以在上面添加关键帧的等级和数值,然后点击缩放匹配
在这里插入图片描述
添加完成数值后,我们可以右键点击自动
在这里插入图片描述
它就可以自动平滑处理
在这里插入图片描述
在这种情况下,你没必要每个等级都设置,并且可以看到对应等级的值
在这里插入图片描述
填充完数据之后,我们将对应职业的属性设置上,在选择使用表格后,前面的数值将是得到的值的倍率,然后选择表格中的曲线,即可实现对应的效果。
在这里插入图片描述
后面我们将使用另外两种格式创建另外两个职业的数值,我们不知道它的格式是怎么样的,可以通过将当前创建的曲线表格导出CSV或者JSON格式
在这里插入图片描述
CSV文件打开就是表格的形式,我们可以以此为模版进行修改,但是CSV格式导入到UE无法进行平滑处理
在这里插入图片描述
CSV格式也可以通过文本打开
在这里插入图片描述

JSON格式就是标准的JSON字符串显示,我们可以通过修改KEY和Value实现值的配置。
在这里插入图片描述

从外部文件导入表格数据

上面我们查看了对应的格式,我们可以以此为模版修改数据,并且直接导回到UE内。
导入到UE分为两种形式,第一种是直接将文件导入,我们可以在内容浏览器选择导入,直接将所需的文件导入到UE内
在这里插入图片描述
选择文件后,需要选择数据表格式和数据表插值类型。
在这里插入图片描述
导入后,如果选的是Cubic,那可以实现光滑取消的效果
在这里插入图片描述
如果你是先创建的文件,然后重新导入
在这里插入图片描述
曲线就会变成线性插值
在这里插入图片描述

使用数据表格设置GE

上面我们实现了使用外部数据创建GE使用的数据表格,这里再仔细讲一下如何使用数据表格设置GE。在你设置属性时,后面有一个使用曲线表格,
在这里插入图片描述
选择我们所使用的表格
在这里插入图片描述
表格里面能够存储多个数据,所以,我们还需要指定使用哪一条曲线
在这里插入图片描述
现在的设置的浮点数将是对从数据表格中获取到的结果的缩放,比如在等级0时,返回5 * 1为最终返回的力量值。
在这里插入图片描述
接下来,我们将所有的属性都设置上去,三个职业对应三个GE。
在这里插入图片描述

实现通过设置的通用GE实例化敌人

创建完成对应的GE,我们并使用数据表格填充了对应的数据,接下来,我们将通过代码实现对应的逻辑去初始化角色。
首先,我们考虑将创建数据放在哪里,这里我们将其放到GameMode上面,对于GameMode来说,在不同场景可以设置不同的GameMode,刚好合适,比如你创建了一个关卡,可以设置上对应的数据,关卡内的敌人初始化时,就使用这一套GE去初始化。
在我们的GameMode基类上增加一个变量,用于设置数据

public:

	UPROPERTY(EditDefaultsOnly, Category="Character Class Defaults")
	TObjectPtr<UCharacterClassInfo> CharacterClassInfo;

接下来,就是如何使用这个数据,由于它是属于通用的函数,我们将应用的函数创建到函数库里面,之前刚好有个自定义的函数库,我们在其内部增加对应的实现函数
在函数库增加一个函数,用于初始角色属性

	//初始化角色的属性
	UFUNCTION(BlueprintCallable, Category="MyAbilitySystemLibrary|CharacterClassDefaults")
	static void InitializeDefaultAttributes(const UObject* WorldContextObject, ECharacterClass CharacterClass, float Level, UAbilitySystemComponent* ASC);

在函数实现中,首先获取到关卡的GameMode,如果类型不一致,获取到的是空指针,将不执行后续操作

	//获取到当前关卡的GameMode实例
	const AMyGameModeBase* GameMode = Cast<AMyGameModeBase>(UGameplayStatics::GetGameMode(WorldContextObject));
	if(GameMode == nullptr) return;

然后获取到我们配置的DataAsset

	//从实例获取到关卡角色的配置
	UCharacterClassInfo* ClassInfo = GameMode->CharacterClassInfo;

从DataAsset获取到配置的对应职业的数据

	//获取到默认的基础角色数据
	const FCharacterClassDefaultInfo ClassDefaultInfo = ClassInfo->GetClassDefaultInfo(CharacterClass);

接下来就是三连应用,和角色一致,先应用主要属性,这个会通过等级设置的不同的属性值,然后根据基础属性生成次级属性的值,最后根据属性填充血量和蓝量的值。

	//应用基础属性
	FGameplayEffectContextHandle PrimaryContextHandle = ASC->MakeEffectContext();
	PrimaryContextHandle.AddSourceObject(WorldContextObject);
	const FGameplayEffectSpecHandle PrimarySpecHandle = ASC->MakeOutgoingSpec(ClassDefaultInfo.PrimaryAttributes, Level, PrimaryContextHandle);
	ASC->ApplyGameplayEffectSpecToSelf(*PrimarySpecHandle.Data.Get());

	//设置次级属性
	FGameplayEffectContextHandle SecondaryContextHandle = ASC->MakeEffectContext();
	SecondaryContextHandle.AddSourceObject(WorldContextObject);
	const FGameplayEffectSpecHandle SecondarySpecHandle = ASC->MakeOutgoingSpec(ClassInfo->SecondaryAttributes, Level, SecondaryContextHandle);
	ASC->ApplyGameplayEffectSpecToSelf(*SecondarySpecHandle.Data.Get());

	//填充血量和蓝量
	FGameplayEffectContextHandle VitalContextHandle = ASC->MakeEffectContext();
	VitalContextHandle.AddSourceObject(WorldContextObject);
	const FGameplayEffectSpecHandle VitalSpecHandle = ASC->MakeOutgoingSpec(ClassInfo->VitalAttributes, Level, VitalContextHandle);
	ASC->ApplyGameplayEffectSpecToSelf(*VitalSpecHandle.Data.Get());

初始化角色属性的函数完成了,接下来,我们要在敌人的基类里面实现使用此函数。
首先在敌人的基类身上增加一个变量,用来设置它的职业

	UPROPERTY(EditAnywhere, BlueprintReadOnly, Category="Character Class Defaults")
	ECharacterClass CharacterClass = ECharacterClass::Warrior;

在敌人基础的角色基础类里,有个函数叫InitializeDefaultAttributes,它是通过在角色的蓝图上面设置的三个GE进行初始化角色,也是当前的主角在使用的方式。
在这里插入图片描述
但是,在敌人这里,我们不需要在角色身上设置对应的GE,而是在GameMode的配置的数据里获取设置,所以,我们将其设置为虚拟函数

	virtual void InitializeDefaultAttributes() const;

在实现这里,只需要调用函数库实现的InitializeDefaultAttributes函数,即可实现对应的效果。

void AEnemyBase::InitAbilityActorInfo()
{
	AbilitySystemComponent->InitAbilityActorInfo(this, this);
	Cast<UAbilitySystemComponentBase>(AbilitySystemComponent)->AbilityActorInfoSet();

	//通过GE初始角色的属性
	InitializeDefaultAttributes();
}

void AEnemyBase::InitializeDefaultAttributes() const
{
	UMyAbilitySystemBlueprintLibrary::InitializeDefaultAttributes(this, CharacterClass, Level, AbilitySystemComponent);
}

编写完成,编译代码打开UE,首先进入GameMode,将我们之前编写的数据设置上去
在这里插入图片描述
然后选择场景中的敌人,设置它的等级和职业
在这里插入图片描述

测试结果

配置完成,现在角色的设置是否成功需要我们去测试结果,我们有多种方式去测试。
第一种,打印数据,在应用完GE后,打印角色的属性值来查看

void AEnemyBase::InitAbilityActorInfo()
{
	AbilitySystemComponent->InitAbilityActorInfo(this, this);
	Cast<UAbilitySystemComponentBase>(AbilitySystemComponent)->AbilityActorInfoSet();

	//通过GE初始角色的属性
	InitializeDefaultAttributes();

	//打印生命值查看属性
	UE_LOG(LogTemp, Warning, TEXT("%s 的生命值为 %f"), *this->GetName(), Cast<UAttributeSetBase>(AttributeSet)->GetHealth());
}

在输出日志里,可以查看到打印的数值
在这里插入图片描述

另一种方法就是打断点,我们将断点打在应用完GE后
在这里插入图片描述
刚好可以查看它的属性值的值是否正确
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/587856.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Unity ParticleSystem 入门

概述 在项目的制作过程成&#xff0c;一定少不了粒子系统的使用吧&#xff0c;如果你想在项目粒子效果&#xff0c;那这部分的内容一定不要错过喔&#xff01;我添加了理解和注释更好理解一点&#xff01; Common Attribute(粒子通用属性) Duration&#xff1a;粒子持续的时间…

分类规则挖掘(二)

目录 三、决策树分类方法&#xff08;一&#xff09;决策树生成框架&#xff08;二&#xff09;ID3分类方法&#xff08;三&#xff09;决策树的剪枝&#xff08;四&#xff09;C4.5算法 三、决策树分类方法 决策树 (Decision Tree) 是从一组无次序、无规则&#xff0c;但有类别…

代谢组数据分析七:从质谱样本制备到MaxQuant搜库

前言 LC-MS/MS Liquid Chromatography-Mass Spectrometry&#xff08;LC-MS/MS &#xff0c;液相色谱-质谱串联&#xff09;可用于残留化合物检测、有机小分子检测、鉴定和定量污染物以及在医药和食品领域添加剂检测和生物小分子等检测。 LC-MS/MS一般包含五个步骤&#xff…

Go Web 开发基础【用户登录、注册、验证】

前言 这篇文章主要是学习怎么用 Go 语言&#xff08;Gin&#xff09;开发Web程序&#xff0c;前端太弱了&#xff0c;得好好补补课&#xff0c;完了再来更新。 1、环境准备 新建项目&#xff0c;生成 go.mod 文件&#xff1a; 出现报错&#xff1a;go: modules disabled by G…

GAI工具哪家强?(ChatGPT 4 vs 文心一言)

开始之前&#xff0c; 先来看看 GAI和AI的区别和关系。 AI 和GAI AI 和GAI的概念 AI&#xff08;Artificial Intelligence&#xff09;是人工智能的缩写&#xff0c;是计算机科学的一个分支&#xff0c;旨在使机器像人类一样进行学习和思考。AI技术的研究领域包括机器人、语…

吴恩达2022机器学习专项课程(一)8.2 解决过拟合

目录 解决过拟合&#xff08;一&#xff09;&#xff1a;增加数据解决过拟合&#xff08;二&#xff09;&#xff1a;减少特征特征选择缺点 解决过拟合&#xff08;三&#xff09;&#xff1a;正则化总结 解决过拟合&#xff08;一&#xff09;&#xff1a;增加数据 收集更多训…

翻译: 什么是ChatGPT 通过图形化的方式来理解 Transformer 架构 深度学习四

合集 ChatGPT 通过图形化的方式来理解 Transformer 架构 翻译: 什么是ChatGPT 通过图形化的方式来理解 Transformer 架构 深度学习一翻译: 什么是ChatGPT 通过图形化的方式来理解 Transformer 架构 深度学习二翻译: 什么是ChatGPT 通过图形化的方式来理解 Transformer 架构 深…

UnityWebGL使用sherpa-ncnn实时语音识别

k2-fsa/sherpa-ncnn&#xff1a;在没有互联网连接的情况下使用带有 ncnn 的下一代 Kaldi 进行实时语音识别。支持iOS、Android、Raspberry Pi、VisionFive2、LicheePi4A等。 (github.com) 如果是PC端可以直接使用ssssssilver大佬的 https://github.com/ssssssilver/sherpa-ncn…

unity入门——按钮点击了却无法调用函数

查阅了一番都没有解决问题&#xff0c;最后发现问题是由button的Onclick()事件绑定了代码脚本而不是游戏对象导致的。 如果Onclick()事件绑定的是代码脚本&#xff0c;则下拉框里没有函数&#xff0c;但是点击MonoScript后能手动填入函数名&#xff08;本以为这样就能实现调用…

使用Python的Tkinter库创建你的第一个桌面应用程序

文章目录 准备工作创建窗口和按钮代码解释运行你的应用程序结论 在本教程中&#xff0c;我们将介绍如何使用Python的Tkinter库创建一个简单的桌面应用程序。我们将会创建一个包含一个按钮的窗口&#xff0c;点击按钮时会在窗口上显示一条消息。 准备工作 首先&#xff0c;确保…

【Python】常用数据结构

1、熟悉字典和列表 2、使用条件判断语句 3、list列表中计算 1、从键盘输人一个正整数列表,以-1结束,分别计算列表中奇数和偶数的和。 &#xff08;1&#xff09;源代码&#xff1a; # 初始化奇数和偶数的和为0 odd_sum 0 even_sum 0 #输入 while True:num int(input(&qu…

java+jsp+Oracle+Tomcat 记账管理系统论文(二)

⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️ ➡️点击免费下载全套资料:源码、数据库、部署教程、论文、答辩ppt一条龙服务 ➡️有部署问题可私信联系 ⬆️⬆️⬆️​​​​​​​⬆️…

分布式链路追踪工具Sky walking详解

1&#xff0c;为什么要使用分布式链路追踪工具 随着分布式系统和微服务架构的出现&#xff0c;且伴随着用户量的增加&#xff0c;项目的体量变得十分庞大&#xff0c;一次用户请求会经过多个系统&#xff0c;不同服务之间调用关系十分复杂&#xff0c;一旦一个系统出现错误都可…

微软如何打造数字零售力航母系列科普06 - 如何使用微软的Copilot人工智能

如何使用微软的Copilot人工智能&#xff1f; Copilot和ChatGPT有很多相似之处&#xff0c;但微软的聊天机器人本身就有一定的优势。以下是如何对其进行旋转&#xff0c;并查看其最引人注目的功能。 ​​​​​​​ &#xff08;资料来源&#xff1a;Lance Whitney/微软&…

C++ | 类和对象(中) (构造函数 | 析构函数 | 拷贝构造函数 | 赋值运算符重载 | 取地址 | const取地址)

目录 默认成员函数 构造函数 构造函数是什么 构造函数特征 什么是默认构造函数 注意事项 编译器自动生成的默认构造 缺省值 对象如何传值给构造函数 初始化列表 析构函数 析构函数的特征 编译器默认生成的析构函数 总结 拷贝构造函数 拷贝构造函数的使用场景 拷…

使用ipxe安装现有的装机环境

iPXE和传统PXE区别 iPXE和传统PXE&#xff08;Pre-boot Execution Environment&#xff0c;预启动执行环境&#xff09;的主要区别在于它们的功能和协议支持。以下是两者的主要区别&#xff1a; 协议支持&#xff1a; PXE仅支持TFTP&#xff08;trivial file transfer protoco…

【linuxC语言】空洞文件

文章目录 前言一、空洞文件1.1 空洞文件的介绍1.2 用途 二、示例代码总结 前言 在 Linux 系统编程中&#xff0c;空洞文件是一种特殊类型的文件&#xff0c;它包含了逻辑上的空洞&#xff0c;也就是说文件中的某些部分并没有实际写入数据。尽管文件在逻辑上可能非常大&#xf…

向eclipse中的项目导入jdk、tomcat

前言&#xff1a; 有些项目无法正常启动可能是因为他的基础配置不正确或者没配置&#xff0c;eclipse中的javaweb项目常见的配置就是jdk、tomcat&#xff0c;这三者配置的方式大概相同&#xff0c;以下是相关操作。我的环境是eclipse2018。 一、jdk 在项目上右键选中propert…

java-Spring-mvc-(请求和响应)

目录 &#x1f4cc;HTTP协议 超文本传输协议 请求 Request 响应 Response &#x1f3a8;请求方法 GET请求 POST请求 &#x1f4cc;HTTP协议 超文本传输协议 HTTP协议是浏览器与服务器通讯的应用层协议&#xff0c;规定了浏览器与服务器之间的交互规则以及交互数据的格式…

Swiper轮播图

版本&#xff1a;“swiper”: “^6.8.4”, 处理每分钟重新请求数据后&#xff0c;播放卡顿&#xff0c;快速闪&#xff0c;没按照设置时间播放等bug 以下是直接vue2 完整的组件代码 使用&#xff1a; <SwiperV :imgList“swiperList” / <template><div class"…