标准C++20无原生编译期反射,需用宏+模板注册字段名、类型、偏移;运行时反射依赖std::any/variant手动维护映射;全自动反射不可行,宏注册是唯一可控路径。
标准 C++20 不提供原生编译期反射,但可通过宏配合模板元编程模拟出“字段名→类型→偏移”的静态映射。关键不是真的反射,而是让编译器在编译时就生成可查的结构描述。
常见错误是试图用 decltype 或 std::is_same 直接推导成员名——C++ 语法不支持运行时获取成员标识符字符串。必须靠宏展开把名字“写死”进类型系统。
REFLECT_STRUCT,在结构体声明后调用,如:REFLECT_STRUCT(Person, name, age)
template 特化方式为每个字段生成唯一类型标签(如 field_tag),并绑定 offsetof 偏移和 std::type_identity_t
#name 转为字面量,存入 constexpr std::string_view 数组(C++20 起支持)struct Person {
std::string name;
int age;
};
REFLECT_STRUCT(Person, name, age); // 展开后生成静态元数据若需要真正动态访问(比如从 JSON 字符串构造对象),就得放弃纯编译期方案,改用运行时注册表。这不是“语言级反射”,而是你自己实现的、带类型擦除的字段访问器。
容易踩的坑是裸指针悬挂或类型转换不安全——void* 偏移加法 + reinterpret_cast 极易出错,必须配对验证。
register_type()
reg.field("name", [](const void* p) { return &static_cast(p)->name; }, ...)
std::any 或自定义 variant,避免裸 void*
type
_info 在不同模块可能不一致auto obj = std::make_unique(); auto& meta = get_reflection_meta (); meta.set_field(*obj, "age", 25); // 内部做类型检查与赋值 std::cout << std::any_cast (meta.get_field(*obj, "age")) << "\n";
没有它们,你就得自己写类型擦除容器或硬编码支持类型列表。C++17 的 std::any 已足够应对多数配置/序列化场景,但注意它不提供运行时类型比较(any.type() == typeid(int) 可用,但不能比 std::type_info 地址)。
性能上,std::any 构造/析构有堆分配开销;若字段全是 POD 类型,可改用 std::variant 避免分配,但需提前穷举所有可能类型。
dynamic_cast 替代 std::any_cast——前者只适用于多态类,且开销更大std::any 存储引用需包装成 std::reference_wrapper,否则会拷贝有人尝试用 template 推导结构体字段,但这根本不可行:C++ 不允许将非类型模板参数用于成员指针(&T::field 不是字面量),也无法从 sizeof(T) 反推字段布局。
所有声称“零宏全自动反射”的库,底层必然依赖编译器扩展(如 Clang 的 __reflect)、外部代码生成(如 protobuf 插件)、或限制极严的 POD 结构体 + 字节解析。标准 C++ 下,宏注册仍是唯一可控路径。
真正难的不是注册字段,而是让反射信息能被序列化器、GUI 绑定、脚本桥接等下游模块一致消费——这意味着你要设计一套稳定的元数据 ABI,而不是每次换种用途就重写一遍映射逻辑。