C++虽无原生完整反射,但可通过typeid和type_info获取基础类型信息,并利用静态初始化、模板与函数指针实现简易反射系统;通过TypeRegistry注册类名与构造函数映射,支持按名称创建对象;结合offsetof和Property结构可注册并访问类的属性信息,适用于序列化、配置加载等场景。

在C++中,原生并不像Java或C#那样支持完整的反射机制,但可以通过一些技巧实现一个简单的运行时类型信息(RTTI)扩展系统。这个系统可以支持基本的类名查询、属性访问和方法调用模拟,适用于配置加载、序列化、调试等场景。
C++标准提供了有限的运行时类型信息支持,主要通过 typeid 和 type_info 实现。
示例:#include <typeinfo><br>class Base {<br> virtual ~Base() = default;<br>};<br><br>class Derived : public Base {};<br><br>Base* obj = new Derived();<br>std::cout << typeid(*obj).name() << std::endl; // 输出类似 "7Derived"<br>std::cout << typeid(*obj).hash_code() << std::endl;注意:typeid 需要目标类型有至少一个虚函数(即多态类型),否则无法正确识别动态类型。
立即学习“C++免费学习笔记(深入)”;
由于C++编译期决定一切,我们可以利用静态初始化机制,在程序启动时注册类信息到全局管理器。
核心思路:
基础实现:
<pre class="brush:php;toolbar:false;">#include <map><br>#include <string><br>#include <functional><br><br>class TypeRegistry {<br>public:<br> template<typename T><br> void register_type(const std::string& name) {<br> creators[name] = []() -> void* { return new T(); };<br> }<br><br> void* create(const std::string& name) {<br> if (creators.find(name) != creators.end()) {<br> return creators[name]();<br> }<br> return nullptr;<br> }<br><br> static TypeRegistry& instance() {<br> static TypeRegistry reg;<br> return reg;<br> }<br><br>private:<br> std::map<std::string, std::function<void*()>> creators;<br>};使用示例:
class MyClass {<br>public:<br> MyClass() { std::cout << "MyClass created\n"; }<br> void say_hello() { std::cout << "Hello\n"; }<br>};<br><br>// 全局注册(可在.cpp文件中)<br>static bool registered = []{<br> TypeRegistry::instance().register_type<MyClass>("MyClass");<br> return true;<br>}();运行时创建:
<code>void* obj = TypeRegistry::instance().create("MyClass");<br>if (obj) {<br> static_cast<MyClass*>(obj)->say_hello();<br>}可以进一步为类注册字段信息,比如名称、偏移量、类型等,实现简单的属性访问。
例如:
简单属性结构:
<pre class="brush:php;toolbar:false;">struct Property {<br> std::string name;<br> size_t offset;<br> size_t size;<br> std::string type_name;<br>};<br><br>// 示例类<br>class Person {<br>public:<br> int age;<br> std::string name;<br><br> static std::vector<Property> get_properties() {<br> return {<br> {"age", offsetof(Person, age), sizeof(int), "int"},<br> {"name", offsetof(Person, name), sizeof(std::string), "std::string"}<br> };<br> }<br>};这样就可以遍历对象的“公开”字段,用于序列化或打印调试信息。
基本上就这些。虽然不如高级语言的反射强大,但在C++限制下已足够应对许多通用需求。关键是利用模板、函数指针和静态初始化机制补足语言缺失的能力。不复杂但容易忽略细节。
以上就是c++++如何实现一个简单的反射系统_c++运行时类型信息扩展的详细内容,更多请关注php中文网其它相关文章!
c++怎么学习?c++怎么入门?c++在哪学?c++怎么学才快?不用担心,这里为大家提供了c++速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号