单播代理
声明代理
声明宏 | 描述 |
---|
DECLARE_DELEGATE(DelegateTypeName); | 声明无函数参数类型,无函数返回值的单播代理DelegateTypeName | DECLARE_DELEGATE_OneParam(DelegateTypeName, Param1Type); | 声明包含一个函数参数类型,无函数返回值的单播代理DelegateTypeName | DECLARE_DELEGATE_
N
u
m
Num
NumParams(DelegateTypeName, Param1Type, Param2Type, …); | 声明包含
N
u
m
Num
Num个函数参数类型,无函数返回值的单播代理DelegateTypeName
(
T
w
o
<
=
N
u
m
<
=
N
i
n
e
)
(Two <= Num <= Nine)
(Two<=Num<=Nine) | DECLARE_DELEGATE_RetVal(RetValType, DelegateTypeName); | 声明无函数参数类型,有函数返回值的单播代理DelegateTypeName | DECLARE_DELEGATE_RetVal_OneParam(ReturnValueType, DelegateName, Param1Type); | 声明包含一个函数参数类型,有函数返回值的单播代理DelegateTypeName | DECLARE_DELEGATE_RetVal_
N
u
m
Num
NumParam(RetValType, DelegateTypeName, Param1Type, Param2Type, …); | 声明包含
N
u
m
Num
Num个函数参数类型,有函数返回值的单播代理DelegateTypeName
(
T
w
o
<
=
N
u
m
<
=
N
i
n
e
)
(Two <= Num <= Nine)
(Two<=Num<=Nine) |
static DelegateTypeName DelegateName; // 声明代理
常用绑定函数的使用
BindUObject(const UserClass* InUserObject, typename TMemFunPtrType<true, UserClass, RetValType (ParamTypes..., VarTypes...)>::Type InFunc, VarTypes... Vars)
使用一个有一个函数参数并且有返回值的代理作为例子:
首先先要到一个类文件中声明代理,创建代理的一些逻辑:
DECLARE_DELEGATE_RetVal_OneParam(int32, DelegateTypeName, FString);
protected:
DelegateTypeName DelegateName;
我这里是测试,所以就将逻辑实现放在了BeginPlay 函数中:
DelegateName.BindUObject(TestActor, &ATestActor::DelegateTestFunction);
if (DelegateName.IsBound())
{
int32 Result = DelegateName.Execute(TEXT("123!"));
UE_LOG(LogTemp, Warning, TEXT("返回值为: %d"), Result);
}
使用BindUObject 绑定一个函数,这个函数类型的返回值以及参数类型一定要和代理的函数类型及返回值一模一样。 使用Execute 执行这个函数,并用Result 储存返回回来的值。 将Result 进行打印测试结果是否成功。
结果:
后面几个函数现在不太懂怎么去用,就整理一下有什么作用分享给大家~ (后面懂了就来补,一定会补的,因为这是我的学习笔记,当用到这个知识点的时候我就会来看这篇文章,然后对他进行一个修改和整理。)
if(!RawObjectPtr.IsValid())
{
RawObjectPtr = MakeShareable(new RawObject);
DelegateName.BindSP(RawObjectPtr.ToSharedRef(), &RawObject::DelegateTest);
}
if (!RawObjectPtr.IsValid())
{
RawObjectPtr = MakeShareable(new RawObject);
DelegateName.BindRaw(RawObjectPtr.Get(), &RawObject::DelegateTest);
}
这个相对来说比较简单,就小标题名字意思。 DelegateName.BindStatic(&MyClass::StaticMethod);
DelegateName.Unbind();
多播代理
声明宏 | 描述 |
---|
DECLARE_MULTICAST_DELEGATE( DelegateTypeName ); | 声明无函数参数类型,无函数返回值的多播代理DelegateTypeName | DECLARE_MULTICAST_DELEGATE_OneParam( DelegateTypeName, Param1Type ); | 声明包含一个函数参数类型,无函数返回值的多播代理DelegateTypeName | DECLARE_MULTICAST_DELEGATE_
N
u
m
Num
NumParams(DelegateTypeName, Param1Type, Param2Type, …); | 声明包含
N
u
m
Num
Num个函数参数类型,无函数返回值的多播代理DelegateTypeName
(
T
w
o
<
=
N
u
m
<
=
N
i
n
e
)
(Two <= Num <= Nine)
(Two<=Num<=Nine) |
注意:多播委托中函数签名不能使用返回值。
常用绑定函数的使用
函数 | 描述 |
---|
Add() | 将函数委托添加到多播委托的调用列表中; | AddStatic() | 添加原始C++指针全局函数委托; | AddRaw() | 添加原始C++指针委托; | AddSP() | 添加基于共享指针的成员函数委托,共享指针委托保留对对象的弱引用; | AddUObject() | 添加基于UObject的成员函数委托,UObject委托保留对对象的弱引用; | Remove() | 从该多播委托的调用列表中删除函数(委托的顺序可能不会被保留); | RemoveAll() | 从该多播委托的调用列表中删除绑定到直到UserObject的所有函数(委托的顺序可能不会被保留)。 |
广播: 调用Broadcast,但是调用不保证执行顺序的正确性。
动态代理
声明动态委托
代码形式上是和普通代理差不多的,就是多了一个_DYNAMIC 。
DECLARE_DYNAMIC_DELEGATE[_RetVal, …]( FDelegateTypeName )
DECLARE_DYNAMIC_MULTICAST_DELEGATE[_RetVal, …]( FDelegateTypeName )
太多了,而且写到这上面也意义不大,讲一下他的注意点吧。
注意:
- 代理名前需要添加一个
F
F
F)。
- 动态代理在类型后面必须加参数名称。
DECLARE_DYNAMIC_DELEGATE_RetVal_OneParam(int32, FDelegateTypeName, FString, Name); - 动态代理绑定的函数,一定要声明
UFUNCTION() ,因为需要跟随代理被序列化(其他代理均不能使用)。
单播无法在蓝图中绑定,无法使用宏BlueprintAssignable (UPROPERTY 中的修饰符,仅能用于Multicast代理。应显示该属性,以供在蓝图中分配)修饰。
动态委托绑定
辅助宏 | 说明 |
---|
BindDynamic( UserObject, FuncName ) | 用于在动态委托上调用BindDynamic()的辅助宏。自动生成函数命名字符串。 | AddDynamic( UserObject, FuncName ) | 用于在动态组播委托上调用AddDynamic()的辅助宏。自动生成函数命名字符串。 | RemoveDynamic( UserObject, FuncName ) | 用于在动态组播委托上调用RemoveDynamic()的辅助宏。自动生成函数命名字符串。 |
和之前差不多的用法。
执行动态委托
执行函数 | 描述 |
---|
Execute | 不检查其绑定情况即执行一个委托 | ExecuteIfBound | 检查一个委托是否已绑定,如是,则调用Execute | IsBound | 检查一个委托是否已绑定,经常出现在包含 Execute 调用的代码前 |
还有其他的函数就是单播和多播各自使用的那些函数了。
动态代理用于蓝图
例:
这个和之前一样现在一个类中声明动态委托:
DECLARE_DYNAMIC_MULTICAST_DELEGATE_OneParam(FDynDelMul, FString, Name);
UPROPERTY(BlueprintAssignable, BlueprintReadWrite)
FDynDelMul DynDel;
UFUNCTION(BlueprintCallable)
void EndTrigger();
在BeginPlay中绑定和广播。
DynDel.AddDynamic(TestActor, &ATestActor::Function);
if (DynDel.IsBound())
{
DynDel.Broadcast("DynDel");
}
EndTrigger内容:
if (DynDel.IsBound())
{
DynDel.Broadcast("DynDel");
}
TestActor类的函数声明:
UFUNCTION()
void Function(FString Name);
函数定义:
void ATestActor::Function(FString Name)
{
UE_LOG(LogTemp, Warning, TEXT("输入字符串为: %s"), *Name);
}
创建有动态委托类的蓝图子类 在蓝图中可以使用:
现在我们可以在蓝图上面进行我们的操作: 创建一个继承ATestActor的蓝图类。
运行查看输出日志:
|