0

0

如何用c++实现一个简单的反射机制? (编译期与运行时方案)

冰火之心

冰火之心

发布时间:2026-01-20 14:07:50

|

296人浏览过

|

来源于php中文网

原创

标准C++20无原生编译期反射,需用宏+模板注册字段名、类型、偏移;运行时反射依赖std::any/variant手动维护映射;全自动反射不可行,宏注册是唯一可控路径。

如何用c++实现一个简单的反射机制? (编译期与运行时方案)

编译期反射:用宏 + 模板注册字段信息

标准 C++20 不提供原生编译期反射,但可通过宏配合模板元编程模拟出“字段名→类型→偏移”的静态映射。关键不是真的反射,而是让编译器在编译时就生成可查的结构描述。

常见错误是试图用 decltypestd::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 起支持)
  • 避免在类内直接使用宏定义字段——会破坏封装且干扰 IDE 补全;应保持原始结构体干净,仅在外部注册
struct Person {
    std::string name;
    int age;
};
REFLECT_STRUCT(Person, name, age); // 展开后生成静态元数据

运行时反射:用 unordered_map 手动维护字段映射

若需要真正动态访问(比如从 JSON 字符串构造对象),就得放弃纯编译期方案,改用运行时注册表。这不是“语言级反射”,而是你自己实现的、带类型擦除的字段访问器。

容易踩的坑是裸指针悬挂或类型转换不安全——void* 偏移加法 + reinterpret_cast 极易出错,必须配对验证。

立即学习C++免费学习笔记(深入)”;

  • 每个可反射类型需显式调用初始化函数,如 register_type()
  • 字段注册用 lambda 封装读写逻辑:reg.field("name", [](const void* p) { return &static_cast(p)->name; }, ...)
  • 读写接口统一返回 std::any 或自定义 variant,避免裸 void*
  • 禁止跨 DLL 边界传递反射元数据——RTTI 和 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";

std::any / std::variant 是运行时反射的底线依赖

没有它们,你就得自己写类型擦除容器或硬编码支持类型列表。C++17 的 std::any 已足够应对多数配置/序列化场景,但注意它不提供运行时类型比较(any.type() == typeid(int) 可用,但不能比 std::type_info 地址)。

Simplified
Simplified

AI写作、平面设计、编辑视频和发布内容。专为团队打造。

下载

性能上,std::any 构造/析构有堆分配开销;若字段全是 POD 类型,可改用 std::variant 避免分配,但需提前穷举所有可能类型。

  • 别用 dynamic_cast 替代 std::any_cast——前者只适用于多态类,且开销更大
  • std::any 存储引用需包装成 std::reference_wrapper,否则会拷贝
  • 若目标平台不支持 C++17,可用 Boost.Any,但注意其异常行为与标准版略有差异

别碰 constexpr if + template parameter pack 的“自动反射”幻觉

有人尝试用 template 推导结构体字段,但这根本不可行:C++ 不允许将非类型模板参数用于成员指针(&T::field 不是字面量),也无法从 sizeof(T) 反推字段布局。

所有声称“零宏全自动反射”的库,底层必然依赖编译器扩展(如 Clang 的 __reflect)、外部代码生成(如 protobuf 插件)、或限制极严的 POD 结构体 + 字节解析。标准 C++ 下,宏注册仍是唯一可控路径。

真正难的不是注册字段,而是让反射信息能被序列化器、GUI 绑定、脚本桥接等下游模块一致消费——这意味着你要设计一套稳定的元数据 ABI,而不是每次换种用途就重写一遍映射逻辑。

相关专题

更多
json数据格式
json数据格式

JSON是一种轻量级的数据交换格式。本专题为大家带来json数据格式相关文章,帮助大家解决问题。

412

2023.08.07

json是什么
json是什么

JSON是一种轻量级的数据交换格式,具有简洁、易读、跨平台和语言的特点,JSON数据是通过键值对的方式进行组织,其中键是字符串,值可以是字符串、数值、布尔值、数组、对象或者null,在Web开发、数据交换和配置文件等方面得到广泛应用。本专题为大家提供json相关的文章、下载、课程内容,供大家免费下载体验。

533

2023.08.23

jquery怎么操作json
jquery怎么操作json

操作的方法有:1、“$.parseJSON(jsonString)”2、“$.getJSON(url, data, success)”;3、“$.each(obj, callback)”;4、“$.ajax()”。更多jquery怎么操作json的详细内容,可以访问本专题下面的文章。

310

2023.10.13

go语言处理json数据方法
go语言处理json数据方法

本专题整合了go语言中处理json数据方法,阅读专题下面的文章了解更多详细内容。

75

2025.09.10

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

318

2023.08.02

css中float用法
css中float用法

css中float属性允许元素脱离文档流并沿其父元素边缘排列,用于创建并排列、对齐文本图像、浮动菜单边栏和重叠元素。想了解更多float的相关内容,可以阅读本专题下面的文章。

567

2024.04.28

C++中int、float和double的区别
C++中int、float和double的区别

本专题整合了c++中int和double的区别,阅读专题下面的文章了解更多详细内容。

99

2025.10.23

if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

751

2023.08.22

Java JVM 原理与性能调优实战
Java JVM 原理与性能调优实战

本专题系统讲解 Java 虚拟机(JVM)的核心工作原理与性能调优方法,包括 JVM 内存结构、对象创建与回收流程、垃圾回收器(Serial、CMS、G1、ZGC)对比分析、常见内存泄漏与性能瓶颈排查,以及 JVM 参数调优与监控工具(jstat、jmap、jvisualvm)的实战使用。通过真实案例,帮助学习者掌握 Java 应用在生产环境中的性能分析与优化能力。

3

2026.01.20

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
WEB前端教程【HTML5+CSS3+JS】
WEB前端教程【HTML5+CSS3+JS】

共101课时 | 8.4万人学习

JS进阶与BootStrap学习
JS进阶与BootStrap学习

共39课时 | 3.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号