【UE5 C++课程系列笔记】21——弱指针的简单使用

目录

概念

声明和初始化

转换为共享指针

打破循环引用

弱指针使用警告


概念

在UE C++ 中,弱指针(TWeakPtr )也是一种智能指针类型,主要用于解决循环引用问题以及在不需要强引用保证对象始终有效的场景下,提供一种可以获取对象访问权限的方式。与共享指针(TSharedPtr )和共享引用(TSharedRef )不同,弱指针不会增加其所指向对象的引用计数,这意味着它不会对对象的生命周期产生维持作用,即不会阻止对象被销毁。

例如,在一些复杂的对象关系结构中,多个对象之间可能相互引用,如果都使用强引用(如 TSharedPtrTSharedRef ),很容易形成循环引用,导致对象的引用计数永远无法降为 0,从而造成内存泄漏。而弱指针可以在这种情况下参与对象的引用关系构建,避免出现循环引用问题,同时又能在对象仍然存在时,有机会获取到对象的有效访问权限。

在访问弱指针引用的对象前,应使用 Pin 函数生成共享指针。此操作确保使用该对象时其将继续存在。如只需要确定弱指针是否引用对象,可将其与 nullptr 比较,或在之上调用 IsValid

声明和初始化

在如下代码中,体现了弱指针不维持对象生命周期的特点,以及通过 Pin 函数检查对象是否还能获取有效访问权限的用法。

在第16行代码中,使用 MakeShared<FMyStruct>() 创建了一个 FMyStruct 类型的对象,并通过 TSharedRef 来管理这个对象,使得 ObjectOwnerRef 指向新创建的对象。此时,该对象的引用计数被初始化为 1。

第17行代码通过将 ObjectOwnerRef 作为参数传递给 TWeakPtr 的构造函数,创建了一个弱指针 ObjectObserver,使其指向与 ObjectOwnerRef 相同的 FMyStruct 对象。需要注意的是,这个操作并不会增加对象的引用计数,对象的生命周期仍然仅由 ObjectOwnerRef (以及后续可能出现的其他指向该对象的共享指针或共享引用)来维持,ObjectObserver 只是建立了一个对该对象的弱引用关系,用于后续在不影响对象生命周期的情况下尝试获取对对象的访问权限。

第18行代码创建了一个 TSharedPtr 类型的共享指针 ObjectOwnerPtr,并通过赋值操作让它也指向 ObjectOwnerRef 所指向的 FMyStruct 对象。此时,对象的引用计数会从 1(仅由 ObjectOwnerRef 维持时)变为 2。

第19行代码调用 ObjectOwnerPtrReset 函数,这会使得 ObjectOwnerPtr 释放对其所指向对象的强引用,对象的引用计数会相应地减 1。在执行完这行代码后,对象的引用计数变回 1,仅由 ObjectOwnerRef 来维持其生命周期。

第20~23行代码使用了 弱指针ObjectObserverPin 函数来尝试获取一个指向原对象的临时共享指针,以检查对象是否仍然可以被访问。Pin 函数会在对象仍然存在(即对应的引用计数大于 0 )的情况下,返回一个指向该对象的临时 TSharedPtr

转换为共享指针

Pin 函数将创建指向弱指针对象的共享指针。只要共享指针在范围内且引用对象,则该对象将持续有效。

如下代码主要展示了如何将一个由共享引用(TSharedRef )管理的对象转换为可通过弱指针(TWeakPtr )来间接访问的形式,并且演示了通过弱指针的 Pin 操作获取临时共享指针(TSharedPtr ),进而访问对象成员函数 PrintAA 的过程,整体体现了弱指针在不影响对象生命周期管理的情况下,实现对对象的安全访问机制。

打破循环引用

出现循环引用的示例:

首先在"FMyStruct"结构体中定义一个共享指针 HoldPtr,并初始化为 nullptr

然后创建两个 FMyStruct 类型的对象,并通过它们各自包含的 TSharedPtr<FMyStruct> 类型成员变量 HoldPtr 互相指向对方,形成了一个循环引用的结构,如下所示。

此时调用"LoopPtr"会发现,并没有输出析构的日志信息,说明产生了循环引用现象,导致对象的引用计数永远无法降为 0。

为了打破循环引用,我们可以使用弱指针来代替共享指针

编译后运行 结果如下,可以看到对象可以正常析构了:

弱指针使用警告

如不想保证数据对象会持续存在时,弱指针将非常有用,但该属性可能会变得异常危险。在以下情况中请谨慎使用弱指针:

  • **在SetMap中用作键。弱指针可能会在未通知容器的情况下随时无效,因此共享指针或共享引用更适用于充当键。可安全地将弱指针用作数值。

  • 虽然弱指针提供 IsValid 函数,但是检查 IsValid 无法保证对象在任何时间长度内均可持续有效。线程安全共享指针可能会因另一线程上的活动而随时无效,因此使用线程安全共享指针应尤其注意。Pin 返回的共享指针将使对象在代码将其清除或其超出范围前保持活跃状态,因此 Pin 函数是用于检查的首选方法,此类检查会导致取消引用或访问存储对象。

官方文档地址:

https://dev.epicgames.com/documentation/zh-cn/unreal-engine/shared-references-in-unreal-engine?application_version=5.3

相关推荐
libra695612 分钟前
esp32开发笔记之一:esp32开发环境搭建vscode+ubuntu
笔记·vscode·ubuntu
Chrikk19 分钟前
NCCL学习笔记-概念全解
笔记·学习
三次元11123 分钟前
JS中函数基础知识之查漏补缺(写给小白的学习笔记)
开发语言·前端·javascript·笔记·ecmascript·原型模式
Pandaconda36 分钟前
【Golang 面试题】每日 3 题(二十二)
开发语言·笔记·后端·面试·golang·go·channel
Jackilina_Stone4 小时前
【HUAWEI】HCIP-AI-MindSpore Developer V1.0 | 第五章 自然语言处理原理与应用(2 自然语言处理关键技术) | 学习笔记
人工智能·笔记·学习·自然语言处理·hcip·huawei
垂杨有暮鸦⊙_⊙4 小时前
2024年6月英语六级CET6听力原文与解析
笔记·学习·六级
油炸自行车5 小时前
【阅读】认知觉醒
笔记·阅读·读书
田梓燊5 小时前
人机交互复习笔记
笔记·人机交互
LuH11246 小时前
【论文阅读笔记】MoGe: 使用最优训练监督解锁开放域图像的精确单目几何估计
论文阅读·笔记
Spcarrydoinb6 小时前
python学习笔记—15—数据容器之列表
笔记·python·学习