C++ 标准本身不支持运行时反射(如 Java 或 C# 那样通过 type_info 获取字段名、调用方法),但可以用宏 + 静态注册 + 类型擦除组合出一套轻量、可控的“伪反射”机制,适用于序列化、配置绑定、调试打印等场景。
typeid 或 std::type_info
std::type_info::name() 返回的是编译器生成的 mangled 名字(如 "St6vectorIiSaIiEE"),不可读、不可靠、无字段信息;typeid 也无法枚举类成员、获取字段偏移或类型元数据。真正需要的是开发者主动声明的结构化元数据。
关键思路:在类定义内部用宏展开出静态成员函数,返回该类型的字段描述表(std::vector),每个 FieldMeta 包含字段名、类型 ID、内存偏移、getter/setter 函数指针。
常见错误是宏里直接写 this->field —— 这会导致编译失败,因为宏展开时不在成员函数上下文中。正确做法是用 lambda 或函数指针捕获偏移量。
实操建议:
REFLECTABLE,接受字段列表,如 REFLECTABLE((int, age), (std::string, name))
offsetof 计算偏移,配合 static_cast 转成 void* 基址加法实现安全访问typeid,改用枚举(如 TYPE_INT, TYPE_STRING)或 std::type_index,避免 RTTI 依赖std::is_standard_layout_v 为 true),否则 offsetof 行为未定义没有编译器支持,就不存在“自动发现字段”。所有反射能力都依赖显式注册 —— 这不是缺陷,而是可控性的代价。
两种常见注册方式对比:
static const auto& reg = []{ ... }();,利用静态变量初始化顺序触发注册,但跨 TU 可能失效(ODR 问题)ReflectRegistry::Register(&Person::GetMeta); ,启动时集中调用,稳定可靠,适合大型项目consteval + 结构化绑定推导字段,但无法绕过“必须写宏声明”这一前提以下代码实现了字段名遍历和值读取,不含 setter 和嵌套对象,但已覆盖 80% 的配置/序列化需求:
#include#include #include #include #include struct FieldMeta { const char* name; std::type_index type; size_t offset; };
template
class Reflectable { public: static const std::vector & GetFields() { static const std::vector fields = T::kFields; return fields; } }; define REFLECTABLE(...) \
static const std::vectorzuojiankuohaophpcnFieldMetayoujiankuohaophpcn kFields; \ static const std::vectorzuojiankuohaophpcnFieldMetayoujiankuohaophpcn BuildFields() { \ return { __VA_ARGS__ }; \ } \ static_assert(true, "")define FIELD(type, name) { #name, std::type_index(typeid(type)), offsetof(THIS_TYPE, name) }
// 使用示例 struct Person { int age; std::string name;
REFLECTABLE( { "age", std::type_index(typeid(int)), offsetof(Person, age) }, { "name", std::type_index(typeid(std::string)), offsetof(Person, name) } );};
// 手动注册(避免宏中无法推导 THIS_TYPE) const std::vector
Person::kFields = Person::BuildFields(); int main() { Person p{42, "Alice"}; for (const auto& f : Reflectable
::GetFields()) { std::cout reinterpret_cast >( reinterpret_cast >(&p) + f.offset ); } else if (f.type == std::type_index(typeid(std::string))) { std::cout reinterpret_cast >( reinterpret_ cast
>(&p) + f.offset ); } std::cout 注意
offsetof对非标准布局类型(如含虚函数、多继承、private 继承)无效;字符串字段读取依赖std::string内存布局(libc++/libstdc++ 不同版本可能有差异);生产环境应封装reinterpret_cast为安全访问器。
# java
# iis
# ai
# c++
# ios
# stream
# c#
# 作用域
# 为什么
相关文章:
建站之星如何取消后台验证码生成?
广州商城建站系统开发成本与周期如何控制?
贸易公司网站制作流程,出口贸易网站设计怎么做?
宝塔面板创建网站无法访问?如何快速排查修复?
建站之星安装后如何自定义网站颜色与字体?
建站主机数据库如何配置才能提升网站性能?
建站之星会员如何解锁更多建站功能?
如何在云主机上快速搭建网站?
建站VPS配置与SEO优化指南:关键词排名提升策略
云南网站制作公司有哪些,云南最好的招聘网站是哪个?
建站主机默认首页配置指南:核心功能与访问路径优化
成都网站制作价格表,现在成都广电的单独网络宽带有多少的,资费是什么情况呢?
哈尔滨网站建设策划,哈尔滨电工证查询网站?
七夕网站制作视频,七夕大促活动怎么报名?
如何高效利用200m空间完成建站?
个人摄影网站制作流程,摄影爱好者都去什么网站?
,怎么用自己头像做动态表情包?
如何通过FTP空间快速搭建安全高效网站?
利用JavaScript实现拖拽改变元素大小
香港服务器建站指南:外贸独立站搭建与跨境电商配置流程
建站之星2.7模板快速切换与批量管理功能操作指南
北京网站制作公司哪家好一点,北京租房网站有哪些?
郑州企业网站制作公司,郑州招聘网站有哪些?
黑客如何利用漏洞与弱口令入侵网站服务器?
如何快速搭建高效可靠的建站解决方案?
如何在服务器上三步完成建站并提升流量?
宿州网站制作公司兴策,安徽省低保查询网站?
如何在西部数码注册域名并快速搭建网站?
如何快速打造个性化非模板自助建站?
网站制作的方法有哪些,如何将自己制作的网站发布到网上?
如何基于云服务器快速搭建网站及云盘系统?
简历在线制作网站免费,免费下载个人简历的网站是哪些?
如何在云指建站中生成FTP站点?
宝塔建站助手安装配置与建站模板使用全流程解析
盘锦网站制作公司,盘锦大洼有多少5G网站?
视频网站制作教程,怎么样制作优酷网的小视频?
制作网站哪家好,cc、.co、.cm哪个域名更适合做网站?
c++怎么用jemalloc c++替换默认内存分配器【性能】
*服务器网站为何频现安全漏洞?
如何通过商城免费建站系统源码自定义网站主题?
b2c电商网站制作流程,b2c水平综合的电商平台?
江苏网站制作公司有哪些,江苏书法考级官方网站?
大同网页,大同瑞慈医院官网?
建设网站制作价格,怎样建立自己的公司网站?
建站主机CVM配置优化、SEO策略与性能提升指南
如何在七牛云存储上搭建网站并设置自定义域名?
如何在建站之星网店版论坛获取技术支持?
如何在Mac上搭建Golang开发环境_使用Homebrew安装和管理Go版本
制作网站的基本流程,设计网站的软件是什么?
python的本地网站制作,如何创建本地站点?
*请认真填写需求信息,我们会在24小时内与您取得联系。