news 2026/5/1 10:55:28

C++ 运行时反射系统设计与实现(无 RTTI)

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
C++ 运行时反射系统设计与实现(无 RTTI)

一、引言

在很多高级语言(如 C#/Java)中,“反射”是一种内省机制,允许程序在运行时查看和操作对象的结构与行为(如类名、字段、方法等)。而 C++ 作为静态语言,默认并不支持强反射机制

尽管 C++ 提供 RTTI(如typeiddynamic_cast),但其功能非常有限,并且很多高性能或嵌入式环境都禁用了 RTTI。

本文将从零开始,实现一个纯 C++ 的运行时反射系统,不依赖 RTTI,可实现字段注册、类型名称、对象属性遍历、动态构造等能力。


二、反射系统的应用场景

场景用途
编辑器属性面板展示与绑定
脚本接口反射 C++ 对象给 Lua/Python
序列化将对象自动转为 JSON/XML/Binary
插件系统不同模块之间的统一字段解析与操作
网络通信自动化消息结构注册与生成

三、目标功能概览

我们的运行时反射系统应具备以下能力:

  • 注册类型信息(类名、字段)

  • 动态构造对象(new 及析构)

  • 遍历字段,获取/设置值

  • 查询字段类型、名称、偏移量

  • 跨模块兼容性(不依赖 RTTI)


四、核心设计思路

设计三个核心类:

  1. Type:表示一个类的信息

  2. Field:表示类的一个字段(属性)

  3. TypeRegistry:集中管理所有注册的类型

我们通过宏 + 模板封装类信息,在运行时自动注册。


五、字段描述结构 Field

cpp复制编辑struct Field { std::string name; std::string type; size_t offset; template<typename T> T& get(void* instance) const { return *reinterpret_cast<T*>((char*)instance + offset); } };

六、类型信息结构 Type

cpp复制编辑struct Type { std::string name; size_t size; std::function<void*(void)> creator; std::function<void(void*)> deleter; std::vector<Field> fields; void* create() const { return creator(); } void destroy(void* obj) const { deleter(obj); } };

七、注册表 TypeRegistry

cpp复制编辑class TypeRegistry {public: static TypeRegistry& instance() { static TypeRegistry inst; return inst; } void registerType(const std::string& name, const Type& t) { types_[name] = t; } const Type* get(const std::string& name) const { auto it = types_.find(name); return it != types_.end() ? &it->second : nullptr; }private: std::unordered_map<std::string, Type> types_; };

八、反射注册宏与辅助函数

cpp复制编辑#define BEGIN_REFLECT(T) \ namespace { \ struct T##_Reflector { \ T##_Reflector() { \ Type t; \ t.name = #T; \ t.size = sizeof(T); \ t.creator = []() -> void* { return new T(); }; \ t.deleter = [](void* obj) { delete (T*)obj; };#define REFLECT_FIELD(field, type) \ t.fields.push_back(Field{#field, #type, offsetof(T, field)});#define END_REFLECT(T) \ TypeRegistry::instance().registerType(#T, t); \ } \ }; static T##_Reflector global_reflector_##T; }

九、定义一个可反射的类

cpp复制编辑struct Person { std::string name; int age; float height; };BEGIN_REFLECT(Person) REFLECT_FIELD(name, std::string) REFLECT_FIELD(age, int) REFLECT_FIELD(height, float)END_REFLECT(Person)

十、使用反射 API

cpp复制编辑void demo_reflect() { const Type* t = TypeRegistry::instance().get("Person"); if (!t) return; void* obj = t->create(); for (const Field& f : t->fields) { std::cout << "Field: " << f.name << ", Type: " << f.type << std::endl; if (f.name == "age") { f.get<int>(obj) = 30; } } Person* p = (Person*)obj; std::cout << "Person.age = " << p->age << std::endl; t->destroy(obj); }

输出结果:

yaml复制编辑Field: name, Type: std::stringField: age, Type: intField: height, Type: floatPerson.age = 30

十一、扩展方向建议

功能实现方式说明
反射方法(成员函数)可通过std::function<void(void*)>储存
继承关系反射Type 中添加 base_class_name 字段
JSON 自动序列化使用字段遍历 +std::ostringstream
属性读写回调添加 Getter/Setter 函数映射
多模块反射(DLL)用注册表为跨模块插件通信提供基础

十二、与 RTTI 的比较

特性RTTI (typeid)自定义反射系统
获取字段
获取类型名称
动态构造对象❌(需手写工厂)
可控字段注册
支持嵌入式/禁 RTTI

十三、实际应用案例

1. 自动序列化为 JSON

cpp复制编辑std::string to_json(const void* obj, const Type* t) { std::ostringstream oss; oss << "{"; for (size_t i = 0; i < t->fields.size(); ++i) { const auto& f = t->fields[i]; oss << "\"" << f.name << "\":"; if (f.type == "int") oss << f.get<int>(obj); else if (f.type == "float") oss << f.get<float>(obj); else oss << "\"" << f.get<std::string>(obj) << "\""; if (i < t->fields.size() - 1) oss << ","; } oss << "}"; return oss.str(); }

输出示例:

json复制编辑{"name":"Tom","age":30,"height":175.5}

十四、总结

本文实现了一个无 RTTI 的运行时反射系统,覆盖:

  • 字段描述与对象内存偏移

  • 类型注册与查找

  • 对象动态构造与释放

  • 字段遍历与读取设置

  • 实战序列化场景应用

该机制可轻松嵌入游戏引擎、插件框架、可视化工具中,提升系统的模块化与扩展能力。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/5/1 6:47:29

C++ STL 源码剖析与底层实现详解

一、引言标准模板库&#xff08;STL, Standard Template Library&#xff09;是 C 最强大的基础组件之一&#xff0c;包含了常用的数据结构&#xff08;如 vector、map、set&#xff09;与算法。STL 的使用门槛不高&#xff0c;但真正掌握其性能特点和底层实现逻辑&#xff0c;…

作者头像 李华
网站建设 2026/4/28 13:09:51

Docker容器自愈能力实战(健康检查成功配置全解析)

第一章&#xff1a;Docker容器自愈能力概述 Docker 容器的自愈能力是指在容器发生故障、崩溃或异常退出时&#xff0c;系统能够自动检测并采取恢复措施&#xff0c;从而保障服务的高可用性与稳定性。这一能力并非 Docker 原生默认开启的功能&#xff0c;而是通过组合使用容器编…

作者头像 李华
网站建设 2026/5/1 10:37:42

告别信息刷屏!RssHub+cpolar 让你的订阅源随时随地能看

RssHub 的核心功能是给各种网站生成 RSS 订阅源&#xff0c;不管是微博热搜、GitHub 仓库更新&#xff0c;还是豆瓣小组、慕课课程&#xff0c;它都能把内容整理成标准格式&#xff0c;方便用阅读器统一查看。适合需要实时追踪信息的自媒体人、关注代码动态的程序员&#xff0c…

作者头像 李华
网站建设 2026/5/1 8:12:35

认识定性数据和定量数据和描述性统计的常规分析

下面内容摘录自《用R探索医药数据科学》专栏文章的部分内容&#xff08;原文5342字&#xff09;。 2篇3章1节&#xff1a;定量数据的集中趋势描述&#xff0c;文末有众数的自定义函数-CSDN博客 在科研中&#xff0c;很多资料经过整理之后&#xff0c;常常需要进行一系列的统计…

作者头像 李华
网站建设 2026/4/27 9:45:58

【Docker安全监控终极指南】:揭秘Falco实时监控的5大核心应用场景

第一章&#xff1a;Docker安全监控的演进与Falco的崛起 随着容器化技术的广泛应用&#xff0c;Docker已成为现代应用部署的核心组件。然而&#xff0c;其轻量、动态和短暂的特性也带来了新的安全挑战。传统的主机级安全监控工具难以深入容器内部行为&#xff0c;无法有效检测异…

作者头像 李华
网站建设 2026/4/18 3:28:53

系统提示词怎么写?教你设置‘你是一个编程助手’提升准确率

系统提示词怎么写&#xff1f;教你设置“你是一个编程助手”提升准确率 在算法竞赛训练营里&#xff0c;一位学生正盯着屏幕发愁&#xff1a;他把一道经典的动态规划题输入到本地部署的小模型中&#xff0c;结果返回的答案不仅逻辑混乱&#xff0c;甚至连基本的边界条件都没考虑…

作者头像 李华