Loading

C++中自定义事件与委托

image
自定义事件,和委托其实是一类操作。
在蓝图中都表现为红色方块。
自定义事件通过DECLARE_EVENT(ClassName, EventName)来创建一个属于ClassName的EventName事件。可以看出一般是没有参数的,如果蓝图中或者C++代码中自带的BeginPlay.
自定义事件
而委托则是通过DECLARE_DYNAMIC_MULTICAST_DELEGATE_'FourParams'来创建的需要根据参数个数更改FourParams。类似蓝图或者C++代码中自带的OnComponentBeginOverlap,OnComponentHit等等
委托
参考
委托和事件都相当于一个传输消息(触发点),一个接受消息(执行点)

绑定操作

这一步相当于蓝图中那些事件发生后要进行的操作,比如OnComponentHit后为了销毁粒子,那么就需要在蓝图中设置方块,连线。而在C++代码中,则需要对于有这个委托或者事件的Actor进行调用,添加设置的函数。

//对于自定义事件
Actor->EventName.AddUobject(this,&Class::Func);

//对于委托
Actor->EventName.AddDynamic(this,&Class::Func);

具体步骤

对于触发点
编写红色方块

  1. 声明委托或者事件(委托需要定义一个参数列表)
  2. 声明一个委托或者事件对象(相当于实例化一个类)--类似USphereComponent* SphereComp;
  3. 在需要的位置触发事件或者委托(执行或者广播),委托需要传入参数

对于执行点
一般在其他写的类中,即蓝图中红色方块之后的操作

  1. 利用上文绑定一些这些委托需要做的函数
  2. 编写函数原型

稍微综合一点就是,在一个类中实现委托或者事件的声明和调用(广播),在另一个需要操作的类中实现函数并绑定到委托或者事件中。

我们在一个ActorComponent类中声明了一个委托,并在其对应的.cpp文件中进行了广播,这个委托主要处理当生命值发生变化时的一些操作。
image
image
然后我们在Actor中绑定组件,这样在蓝图中就会显示出这个委托
image
接着我们在C++代码中定义了一个函数来进行角色死亡之后的操作
image
注意:要绑定到委托的函数的形参个数必须和委托的形参数量一致
然后利用AddDynamic绑定
image
重新编译后回到蓝图,就可以发现这个委托拥有了函数IsDied
image
而如果不适用绑定,那么就可以在蓝图中从这个委托事件连线,实现c++代码的模块连接就可以了,有点类似于将蓝图模块写好然后利用Collapse to Function使得功能变为一个Function,只不过我们直接在C++中操作了。

那么这样看来委托其实就是函数,但他相对于函数的优点就在于当发生这个事件时,会对Actor产生影响,而我们使用代理相当于时直接调用,不用传入自身self的引用。而我们使用函数,就需要确定这个函数对谁进行了操作。相当于一种解耦。
将一个普适性的东西使用委托,有利于解耦。

分类

委托分为单播,多播,动态,动态多播,事件(可以看到事件其实就是一种委托)

  1. 单播
    在执行点绑定函数时只能绑定一个函数,比如上方示例如果绑定了IsDied就不可以绑定其他函数了,而如果绑定了多个就选择最后一个。
    其一般的绑定方式为Bind*
    可以拥有返回值。
    在参考中有这样一个例子
DECLARE_DELEGATE_OneParam(FTestDelegate, int);
static void StaticDelegateProc(int nCode)
{
    UE_LOG(LogTemp, Log, TEXT("StaticDelegateProc : %d"), nCode);
}
FTestDelegate DelegateObj1;
//绑定函数
DelegateObj1.BindStatic(StaticDelegateProc);
//通过ExcuteIfBound方法执行
DelegateObj1.ExecuteIfBound(1);
  1. 多播
    DECLARE_MULTICAST_DELEGATE_XXXPARAMS
    一般通过Add*来绑定函数,而其不会有返回值,只能通过广播来调用。
    广播Broadcast()其实就是执行函数,这个委托通过Add绑定的所用函数都会被执行,我们再来看之前那个例子(虽然不是)
bool USAttributeComponent::ApplyHealthChange(float Delta)
{
	if (Health <= 100.0f && Delta < 0.0f) {
		Health += Delta;
		OnHealthChanged.Broadcast(nullptr, this, Health, Delta);
	}
	if (Health < 100.0f && Delta > 0.0f) {
		Health += Delta;
		OnHealthChanged.Broadcast(nullptr, this, Health, Delta);
	}
	return true;
}

先对生命进行扣血或者加血,然后调用委托,而现在只绑定了一个函数就是IsDied,相当于就是

if (Health <= 100.0f && Delta < 0.0f) {
	Health += Delta;
	IsDied();
}

如果绑定了多个函数,需要注意执行顺序可能与函数的添加顺序不相同。
多播

  1. 事件
    事件就是一个多播委托。
    image
    需要注意的是事件的Broadcast()只对事件所在的类有效,所以为了在想要触发事件的位置调用事件需要新建一个接口将Broadcast()放入其中。
    事件

  2. 动态委托和动态多播委托
    主要区别就是动态委托可以在蓝图中调用,就是我们的例子,注意动态多播委托只能作为变量给蓝图调用,不能作为函数参数,而动态委托可以作为函数参数
    动态单播:DECLARE_DYNAMIC_DELEGATE_XXXPARAMS
    动态多播:DECLARE_DYNAMIC_MULTICAST_DELEGATE_XXXPARAM
    动态委托

小结

我们使用了动态多播委托,在C++中绑定了函数给委托,并且在ApplyHealthChange函数执行时,广播了这个委托(执行函数),判断当前角色血量这个操作。
那么在蓝图中如何绑定一个函数(事件)给这个动态多播委托呢?
参考
image
可以看到OnMyDelegate2为我们的动态多播委托,而charDelegate为我们要绑定的事件,类似我们写的IsDied函数。
之后继续在委托编写的类中定义了一个触发委托广播的函数TestBpDelegate,在蓝图中使用按键来执行。
image

posted @   XTG111  阅读(237)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· 全程不用写代码,我用AI程序员写了一个飞机大战
· DeepSeek 开源周回顾「GitHub 热点速览」
· 记一次.NET内存居高不下排查解决与启示
· MongoDB 8.0这个新功能碉堡了,比商业数据库还牛
· 白话解读 Dapr 1.15:你的「微服务管家」又秀新绝活了
点击右上角即可分享
微信分享提示