代理:

  • 代理可以帮助我们解决一对一或是一对多的任务分配工作。主要可以帮助我们解决通知问题。我们可以通过代理完成调用某一个对象的一个函数,而不直接持有该对象的任何指针。
  • 代理就是为你跑腿送信的,你可以不用关心给送信的目标人具体是谁,只要按照约定好的信件格式进行送信即可
  • 更简单理解,想去调用某个函数,但并不是直接去调用,而是通过另一个入口去调用(代理)

分类:

  • 单播代理 只能进行通知一个人
  • 多播代理 可以进行多人通知
  • 动态代理 可以被序列化(这体现在于蓝图进行交互,C++中可以将通知事件进行蓝图广播)

单播代理:

通过宏进行构建,单播代理只能绑定一个通知对象,无法进行多个对象通知、

单播代理分为有返回值与无返回值两种

代理可使用声明宏

函数签名

声明宏

void Function()

DECLARE_DELEGATE(DelegateName)

void Function(Param1)

DECLARE_DELEGATE_OneParam(DelegateName, Param1Type)

void Function(Param1, Param2)

DECLARE_DELEGATE_TwoParams(DelegateName, Param1Type, Param2Type)

void Function(Param1, Param2, ...)

DECLARE_DELEGATE_<Num>Params(DelegateName, Param1Type, Param2Type, ...)

<RetValType> Function()

DECLARE_DELEGATE_RetVal(RetValType, DelegateName)

<RetValType> Function(Param1)

DECLARE_DELEGATE_RetVal_OneParam(RetValType, DelegateName, Param1Type)

<RetValType> Function(Param1, Param2)

DECLARE_DELEGATE_RetVal_TwoParams(RetValType, DelegateName, Param1Type, Param2Type)

<RetValType> Function(Param1, Param2, ...)

DECLARE_DELEGATE_RetVal_<Num>Params(RetValType, DelegateName, Param1Type, Param2Type, ...)

常用绑定函数:

  • BindUObject 绑定UObject类型对象成员函数的代理
  • BindSP 绑定基于共享引用的成员函数代理
  • BindRaw 绑定原始自定义对象成员函数的代理,操作调用需要注意执行需要检查
  • IsBound BindStatic 绑定全局函数成为代理
  • UnBind 解除绑定代理关系

注意:绑定中传递的对象类型必须和函数指针所属类的类型相同否则绑定会报错

调用执行:

  • 为了保证调用的安全性,执行Execute函数之前需要检查是否存在有效绑定使用函数、
  • IsBound Execute 调用代理通知,不安全,需要注意
  • ExecuteIfBound 调用代理通知,安全,但是有返回类型的回调函数无法使用此函数执行回调
  • IsBound 检查当前是否存在有效代理绑定

构建步骤:

  • 通过宏进行声明代理对象类型(根据回调函数选择不同的宏)
  • 使用代理类型进行构建代理对象
  • 绑定回调对象,和操作函数
  • 执行代理对象回调
// Actor1.h
// 头文件下
DECLARE_DELEGATE(DelegateOne)
DECLARE_DELEGATE_RetVal_OneParam(int32 ,DelegateTwo, int32)
// 变量声明
class AActor2* ac2; DelegateOne DegOne;
DelegateTwo DegTwo;
// Actor1.cpp
// 这里将代码写在了BeginPlay中,方便测试
ac2 = GetWorld()->SpawnActor<AActor2>(AActor2::StaticClass()); // 绑定无参无返回值单播代理
DegOne.BindUObject(ac2, &AActor2::CallBackNone);
DegOne.ExecuteIfBound();
// 绑定有参有返回值单播代理
DegTwo.BindUObject(ac2, &AActor2::CallBackRes);
int32 num = 0;
num = DegTwo.Execute(100);
UKismetSystemLibrary::PrintString(this, FString::Printf(TEXT("%d"),num)); /////////////////////////////////////////////////////// // Actor2.h
//声明两个被用来绑定的的函数
void CallBackNone();
int32 CallBackRes(int32 num);
// Actor2.cpp
void AActor2::CallBackNone()
{
UKismetSystemLibrary::PrintString(this, TEXT("无返回值无参数函数调用!"));
} int32 AActor2::CallBackRes(int32 num)
{
UKismetSystemLibrary::PrintString(this, TEXT("有返回值有参数函数调用!"));
return num;
}

测试结果:

 多播代理:

无法构建具有返回值的多播代理——多播代理无返回值

DECLARE_MULTICAST_DELEGATE[_Const, _RetVal, _etc.] (DelegateName)

多播代理绑定函数 

函数

说明

"Add()"

将函数委托添加到该多播委托的调用列表中。

"AddStatic()"

添加原始C++指针全局函数委托。

"AddRaw()"

添加原始C++指针委托。原始指针不使用任何类型的引用,因此如果从委托下面删除了对象,则调用此函数可能不安全。调用Execute()时请小心!

"AddSP()"

添加基于共享指针的(快速、非线程安全)成员函数委托。共享指针委托保留对对象的弱引用。

"AddUObject()"

添加基于UObject的成员函数委托。UObject委托保留对对象的弱引用。

"Remove()"

从该多播委托的调用列表中删除函数(性能为O(N))。请注意,委托的顺序可能不会被保留!

"RemoveAll()"

从该多播委托的调用列表中删除绑定到指定UserObject的所有函数。请注意,委托的顺序可能不会被保留!

广博: 调用函数Broadcast,但是调用不保证执行顺序的正确性

构建步骤:

  • 使用宏构建代理类型
  • 使用代理类型构建多播代理对象
  • 添加绑定代理
  • 执行调用

多播代理执行使用的是 Broadcast() 进行执行函数

动态代理:

  • 允许被序列化的数据结构,这将使得代理可以被数据化提供给蓝图进行使用,达到在CPP中调用代理广播,事件通知到蓝图中。
  • 动态代理和普通代理基本相同,分为单向和多向,动态代理无法使用带有返回值的函数进行构建(动态单播除外,并且单播无法在蓝图中绑定无法使用宏BlueprintAssignable修饰)
  • UE中的大部分通知事件均使用动态代理(方便蓝图操作),如碰撞通知

动态单播代理:

  • DECLARE_DYNAMIC_DELEGATE[_Const, _RetVal, etc.]( DelegateName )

动态多播代理:

  • DECLARE_DYNAMIC_MULTICAST_DELEGATE[_Const, _RetVal, etc.]( DelegateName )

操作函数:

  • BindDynamic( UserObject, FuncName ) 在动态代理上调用BindDynamic()的辅助宏。
  • AddDynamic( UserObject, FuncName ) 在动态多播代理上调用AddDynamic()的辅助宏。
  • RemoveDynamic( UserObject, FuncName ) 在动态多播代理上调用RemoveDynamic()的辅助宏。

与单播多播区别:

  • 动态代理构建类型名称需要用 F 开头(动态代理实现机制构建了类)
  • 动态代理对象类型可以使用UPROPERTY标记,其他代理均无法使用(不加编译可过,调用出错)
  • 动态代理绑定对象的函数需要使用UFUNCTION进行描述(因为需要跟随代理被序列化)

构建:

// Actor1.h
DECLARE_DYNAMIC_DELEGATE(FDelegateTree); // 注意分号
// 变量定义
class AActor2* ac2;
FDelegateTree DegTree;
// Actor1.cpp
ac2 = GetWorld()->SpawnActor<AActor2>(AActor2::StaticClass()); DegTree.BindDynamic(ac2, &AActor2::CallBackNone);
if (DegTree.IsBound()) {
DegTree.ExecuteIfBound();
} //////////////////////////////////////////////////////////
// Actor2.h
UFUNCTION()
void CallBackNone();
// Actor2.cpp
void AActor2::CallBackNone()
{
UKismetSystemLibrary::PrintString(this, TEXT("无返回值无参数函数调用!"));
}

测试结果:

 动态代理用于蓝图:

在构建动态代理提供蓝图使用时,需要在代理上增加标记宏UPROPERTY(BlueprintAssignable)

UEC++ 代理/委托的更多相关文章

  1. UEC 利用代理/委托写一个生命组件

    首先基于ActorComponent创建一个组件 HealthComponent,将需要的变量与函数创建 #include "CoreMinimal.h" #include &qu ...

  2. 代理委托和block

    delegate 委托是协议的一种,顾名思义,就是委托他人帮自己去做事.委托是给一个对象提供机会对另一个对象中的变化做出反应或者影响另一个对象的行为.其基本思想是:两个对象协同解决问题,并且打算在广泛 ...

  3. 【C#】事件(Event)和代理/委托(Delegate)

    代理(Delegate)的例子 delegate void MyDelegate(string str,int index); // 声明代理 class Test { public static v ...

  4. JavaScript中的事件代理/委托

    事件委托在JS高级程序设计中的定义为"利用事件冒泡,只指定一个事件处理程序,就可以管理某一类型的所有事件" 如何理解上面的这句话呢,在网上,大牛们一般都使用收快递这个例子来解释的, ...

  5. C# 代理/委托 Delegate

    本文转载自努力,努力,努力 1. 委托的定义:委托是函数的封装,它代表一"类"函数.他们都符合一定的签名:拥有相同的参数列表,返回值类型.同时,委托也可以看成是对函数的抽象,是函数 ...

  6. iOS delegate, 代理/委托与协议.

    之前知知道iOS协议怎么写, 以为真的跟特么java接口一样, 后来发现完全不是. 首先, 说说应用场景, 就是当你要用一个程序类, 或者说逻辑类, 去控制一个storyboard里面的label, ...

  7. js中事件代理(委托)

    var oul = document.getElementById(‘uli’); oul.onclick = function(e) { e = e || window.event; var tar ...

  8. [置顶] Objective-C ,ios,iphone开发基础:protocol 协议(委托,代理)的声明

    协议是为了弥补Objective-c中类只能单继承的缺陷,在Objective-c2.0之前当一个类遵循一个协议的时候,必须在类中实现协议的所有方法,在Objective-c2.0之后协议中的方法就有 ...

  9. OC 代理 协议 委托 数据源的概念

    (网摘) OBJC 中的 protocol 相当于 java 里的接口,delagate 就是接口的实现类(C中的回调类似 ): 数据源就是对象遵循了存储数据的协议,可以存储使用数据 协议表示了方法可 ...

随机推荐

  1. NC15163 逆序数

    NC15163 逆序数 题目 题目描述 在一个排列中,如果一对数的前后位置与大小顺序相反,即前面的数大于后面的数,那么它们就称为一个逆序.一个排列中逆序的总数就称为这个排列的逆序数.比如一个序列为 \ ...

  2. StarGAN论文及代码理解

    StarGAN的引入是为了解决多领域间的转换问题的,之前的CycleGAN等只能解决两个领域之间的转换,那么对于含有C个领域转换而言,需要学习C*(C-1)个模型,但StarGAN仅需要学习一个,而且 ...

  3. 牛客SQL刷题第三趴——SQL大厂面试真题

    01 某音短视频 SQL156 各个视频的平均完播率 [描述]用户-视频互动表tb_user_video_log.(uid-用户ID, video_id-视频ID, start_time-开始观看时间 ...

  4. 开源数据质量解决方案——Apache Griffin入门宝典

    提到格里芬-Griffin,大家想到更多的是篮球明星或者战队名,但在大数据领域Apache Griffin(以下简称Griffin)可是数据质量领域响当当的一哥.先说一句:Griffin是大数据质量监 ...

  5. 我为 Netty 贡献源码 | 且看 Netty 如何应对 TCP 连接的正常关闭,异常关闭,半关闭场景

    欢迎关注公众号:bin的技术小屋,本文图片加载不出来的话可查看公众号原文 本系列Netty源码解析文章基于 4.1.56.Final版本 写在前面..... 本文是笔者肉眼盯 Bug 系列的第三弹,前 ...

  6. shell中各种括号的用法

    一.单小括号()1.将某个命令的返回值作为某个变量的值进行传递 #!/bin/bash USER=$(whoami) echo $USER [root@jump ~]# for i in $(seq ...

  7. 第九天python3 闭包、nonlocal、默认值的作用域

    闭包 自由变量:未在本地作用域中定义的变量,例如定义在内存函数外的外层函数的作用域中的变量: 闭包:出现在嵌套函数中,指的是内层函数引用到了外层函数的自由变量,就形成了闭包: 示例1: # -*- c ...

  8. 什么是 Base64 ?

    Base64 是什么? Base64是一种二进制到文本的编码方式.如果要更具体一点的话,可以认为它是一种将 byte数组编码为字符串的方法,而且编码出的字符串只包含ASCII基础字符,就是包括小写字母 ...

  9. Docker非root用户使用

    Docker 用户管理 安装Docker后docker相关命令都需要加上sudo才能执行,这里为特定用户添加下权限 Docker群组 不过一般安好docker后该群组已创建 sudo groupadd ...

  10. 2510-Druid监控功能的深入使用与配置-基于SpringBoot-完全使用java config的形式

    环境 springboot 1.5.9.RELEASE + JDK1.8 配置步骤 分两步,1 配置数据源 2 配置监控 直接上代码 1 配置数据源 package com.company.proje ...