尧图网站建设 尧图网络
  • 首页
  • 关于我们
  • 服务项目
  • 案例展示
  • 建站流程
  • 资讯中心
  • 联系我们
首页/资讯中心/详情

《Unreal 对 C++ 做了什么》系列 04. USTRUCT 与 UPROPERTY:数据结构的反射化与变量管理

《Unreal 对 C++ 做了什么》系列 04. USTRUCT 与 UPROPERTY:数据结构的反射化与变量管理
📅 发布时间:2026/6/19 1:44:04

《Unreal 对 C++ 做了什么》系列 (04/54)

04. USTRUCT 与 UPROPERTY:数据结构的反射化与变量管理 💎(深挖版)

🚀 导言:为什么裸 C++ 变量在 UE 里是“瞎子”?

在标准 C++ 中,如果你定义了int32 Health;,编译器只会在内存里留出 4 个字节。程序运行起来后,除了你的代码,没有人知道这 4 个字节代表“血量”。

虚幻引擎通过USTRUCT和UPROPERTY为这段内存贴上了**“身份标签”**。


🔑 USTRUCT:伪装成类的数据包

虽然USTRUCT在 C++ 层面上是个结构体,但在 UHT 看来,它是一个**“被剥夺了部分权力的 UClass”**。

1. 它没有“户口” (No Path)

UObject都有一个在引擎内部的路径(如/Game/Maps/Level1.ActorA),但USTRUCT没有。它必须寄生在某个UObject的内存块里,或者作为一个局部变量存在。

2. 内存对齐与反射

当你在USTRUCT里写下GENERATED_BODY()时,UHT 会生成一个StaticStruct()函数。

  • 标准 C++ 痛点:你无法通过变量名字符串(如 “Health”)去访问结构体成员。
  • UE 解决方案:UScriptStruct。每一个USTRUCT都有一个对应的UScriptStruct元数据对象。它记录了结构体的大小、对齐方式以及每个成员的偏移量。

⚡ UPROPERTY:内存的“监控探针”

这是虚幻 C++ 的灵魂宏。我们将它拆解为三个层面:

1. 自动清空机制(生命周期的保护神)

这是新手最容易忽视的一点。

  • 裸指针:AActor* MyActor;。如果MyActor在关卡里被销毁了,你的指针依然指向原来的地址(野指针),访问即崩溃。
  • UPROPERTY 指针:UPROPERTY() AActor* MyActor;。当该 Actor 被销毁时,UE 会遍历所有引用它的UPROPERTY指针,并**强行将其设为nullptr**。这就是为什么 UE 推荐你访问指针前先做if(MyActor)判断。
2. 说明符(Specifiers)的深层逻辑

说明符不是注释,它们直接改变了 UHT 生成的代码逻辑:

  • **EditAnywherevsEditInstanceOnly**:

  • EditAnywhere:在蓝图类(原型)和场景实例中都能改。

  • EditInstanceOnly:只能在摆放到场景里的那个“分身”上改。这在内存上涉及到CDO (Class Default Object)的数据覆盖机制。

  • BlueprintReadOnly:UHT 会在生成的代码中,拒绝为该变量生成“写入”权限的反射接口,即使你在 C++ 里它是public的。

3. 内存布局的“黑盒”访问

UE 的编辑器面板(Details Panel)其实是一个“内存编辑器”。
当你拖动滑动条修改血量时,流程如下:

  1. 编辑器通过反射找到该变量的FProperty对象。
  2. FProperty存储了相对于类起始位置的Offset(偏移量)。
  3. 编辑器计算:Address = (Byte*)ObjectInstance + Offset。
  4. 直接往这个内存地址里写值。
    这就是为什么即使你的变量是private的,只要加了UPROPERTY(EditAnywhere),编辑器依然能改它——因为它绕过了 C++ 访问权限,直接操作内存。

🛠️ 进阶:TArray/TMap 与 UPROPERTY 的化学反应

如果你的容器里装的是UObject*,必须加UPROPERTY。

// 错误:GC 扫描不到容器内部,里面的 Actor 随时会被回收,留下满地野指针TArray<AActor*>MyActors;// 正确:UE 的 GC 系统会递归扫描容器内部的每一个元素,保护它们的生命周期UPROPERTY()TArray<AActor*>MyActors;

⚠️ 总结:UE 对你的变量做了什么?

特性标准 C++虚幻 C++ (UPROPERTY)
存在感编译后消失运行时全局可查 (Reflection)
安全性需手动置空,否则野指针自动置空 (Auto-nulling)
可达性只能通过代码访问编辑器、蓝图、序列化系统均可访问
内存成本只有变量本身大小变量大小 + 反射系统中的元数据开销

结语

在 UE 中,UPROPERTY是你和引擎之间的契约。你通过宏告诉引擎:“请帮我看着这个变量”。作为回报,引擎赋予了它在编辑器里变现、在网络里同步、以及在 GC 中幸存的能力。


下一篇我们将深入探讨:《05. UFUNCTION:让函数在引擎内可见 (RPC, Exec, Blueprint)》,我们将看看 UE 如何让一个 C++ 函数跨越语言界限,被蓝图甚至网络另一端的电脑调用。

相关新闻

  • Skyvern终极指南:5分钟掌握AI自动化神器,快速实现业务流程自动化
  • 会展设计公司哪家经验丰富?行业内值得关注的服务案例 - 品牌排行榜
  • 就这个人物一致性,我宣布GPT Image 1.5无敌了,秒杀Nano Banana Pro

最新新闻

  • 3步解锁老旧Mac新生命:OpenCore Legacy Patcher终极升级指南
  • 2026宜昌非急救转运救护车TOP5盘点|宜荆荆同城、长江跨江、三峡山地、院区转诊首选康跃转运 - 吉修匠
  • 2026年湖北百合种植基地推荐排行榜:百合技术/百合回收/百合种苗案例参考 - 新闻快传
  • 告别龟速与超时:全方位解决 git clone 网络难题的实战指南
  • 嵌入式MCU电气特性与FLASH操作深度解析:从数据手册到稳定设计
  • 2026 郑州八大装修公司综合实力排行榜 - GrowthUME

日新闻

  • 5分钟掌握Python进化算法:Geatpy高性能优化工具完全指南
  • Microchip 24AA044 EEPROM选型与应用全指南:从参数解析到实战编程
  • 华为的鸿蒙到底有多牛?为什么称作遥遥领先?

周新闻

  • 3步解锁iOS设备:applera1n激活锁绕过完全指南
  • 39 2026 人工智能证书终极盘点,普通人选 AI 证书可以从这些方向入手
  • Redis 暴露公网有多危险?从端口检查到补救步骤

月新闻

  • 【总结】入门篇:50句话让你记住架构核心概念
  • WeChatMsg技术方案解析:实现Mac微信数据自主管理的完整解决方案
  • WeChatMsg:革新性微信数据备份方案,打造你的专属数字记忆库

关于尧图

  • 公司简介
  • 团队介绍
  • 企业文化
  • 荣誉资质

服务项目

  • 定制开发
  • 电商建站
  • UI 设计
  • 运维服务

快速链接

  • 案例展示
  • 建站流程
  • 常见问题
  • 资讯中心

联系方式

  • 📍北京市朝阳区互联网产业园 A 座 10 层
  • 📞400-888-8888
  • ✉️contact@rkmt.cn
  • 🕐周一至周日 9:00-21:00

© 2024 北京尧图网络科技有限公司 版权所有 | 京 ICP 备 XXXXXXXX 号