我有一个struct
和其他一些struct
s 作为成员。外部和内部结构都是StandardLayout
(甚至可以假设内部是普通的旧数据)。像这样的东西:
struct Inner1 {
int a = 0, b = 0;
};
struct Inner2 {
int c = 0, d = 0;
};
struct Outer {
Inner1 x;
std::string s;
Inner2 y;
};
我想编写一些函数,该函数采用Outer&
某种类型的对象T
,可以返回任何嵌套字段的值,具体取决于参数:
int get(Outer& o, T field);
如果Outer
是一个平面结构,指向成员的指针正是我所需要的,但它不是平面的。
最简单的方法是对所有字段进行T
aenum
并编写 a switch
,但它效率不高。更快的方法是进行T
偏移并写入
int get(Outer& o, size_t field) {
return *reinterpret_cast<int*>(reinterpret_cast<char*>(o) + field);
}
然后像get(o, offsetof(Outer, y) + offsetof(Inner2, c))
. 它可以工作,但我不确定它是否可以保证工作 - 对这样的偏移量求和是否正确,以及通过偏移量获取成员值是否安全。
那么问题来了:这种方式安全吗?如果没有,有没有安全的方法?构造的值T
可以是任意复杂的,但是使用它们应该很快。
动机:我需要以某种顺序放置一些嵌套字段中的值,在启动时已知,但在编译时不知道。我想T
在启动时创建一个数组,然后在获取特定对象时使用这个预先计算的数组。
[UPD]:所以它会这样使用:
void bar(int);
void foo(Outer& o, vector<T>& fields) {
for (auto& field : fields) {
bar(get(o, field));
}
}