C++无内置完整反射机制,但可通过RTTI、手动元数据注册及模板宏技巧实现有限反射:利用typeid和dynamic_cast支持类型查询与安全转型,结合TypeDesc结构体与宏注册字段/方法,辅以constexpr和std::tuple提升编译期自动化程度。
C++ 本身没有内置的、类似 Java 或 C# 那样完整的反射机制(比如通过字符串名获取类、调用任意成员函数、枚举字段等),但可以通过 RTTI(Run-Time Type Information) + 手动元数据注册 + 模板与宏技巧 在运行时实现有限但实用的反射能力。核心思路是:让类型“自我描述”,并在程序启动时构建一张可查询的类型信息表。
RTTI 是 C++ 标准提供的最小反射支持,仅限于多态类型(含虚函数的类):
typeid(obj) 返回 std::type_info&,可获取类型名称(.name(),注意该名未标准化,通常为 mangled 名);dynamic_cast(ptr) 实现安全的向下转型,依赖虚函数表中的 RTTI 数据;主流方案(如 Unreal Engine、Qt 的 moc、或开源库 phoenix)都采用“声明即注册”模式:
struct TypeDesc,包含类名、父类指针、构造函数指针、字段列表(每个字段含名字、偏移、类型 ID)、方法列表等;REFLECT_CLASS(MyClass))在类定义后自动生成静态初始化代码,将 TypeDesc 注册到全局哈希表(如 std::unordered_map<:string const typedesc>);offsetof 和模板推导类型,例如:一旦元数据就位,就能写出通用逻辑:
auto obj = Reflection::Create("MyClass"); // 返回 std::unique_ptr;Reflection::SetField(obj.get(), "age", 42);;
int x = Reflection::GetField(obj.get(), "age");
std::function 包装成员函数),支持反射式调用 "SetName" 并传参。C++17/20 可进一步降低人工维护成本:
constexpr 字符串和 if constexpr 在编译期生成字段列表(避免宏);std::tuple 和 std::index_sequence 自动遍历 POD 成员(需用户显式继承某个反射基类或使用 ADL 探测);-Xclang -ast-dump 或第三方工具(如 foonathan::compatibility)可辅助生成反射代码,但非标准流程。基本上就这些——C++ 反射不是开箱即用的功能,而是靠设计约定 + 工程取舍实现的“可控反射”。它不复杂但容易忽略初始化顺序、跨模块符号可见性、以及调试时 mangled 名带来的坑。真正落地时,建议从字段序列化(如 JSON 绑定)这类明确场景切入,再逐步扩展。