【问题标题】:Can I manually access fields by their raw offset in C++?我可以通过 C++ 中的原始偏移量手动访问字段吗?
【发布时间】:2023-03-24 08:31:02
【问题描述】:

以下 sn-p 是否使用未定义/未指定/等。行为?

#include <cstddef>
#include <iostream>
#include <string>

class Test {
    std::string s1{"s1"}, s2{"s2"};
    std::ptrdiff_t offset = (char*)(&s2) - (char*)(this);
public:
    std::string& get() { return *(std::string*)((char*)(this) + offset); }
};

int main() {
    Test test;
    std::cout << Test{test}.get(); // note the copy
}

offset 的目的是指向s1s2(在运行时选择)并且不包含用于复制/移动/访问的特殊逻辑。 std::string 这里只是一个非平凡的类的例子。

【问题讨论】:

  • 我不明白您要解决什么问题。如果无论如何您都有一个“哨兵”值,只需将该哨兵(您的偏移成员)本身设为一个指针,直接指向您要访问的内容。否则,这不只是对offsetof 宏的奇怪重复,可能不是正确的指针算术吗?
  • 这里有 offsetof 宏。但它仅对非标准布局的类型有条件地支持(并且您的类不是标准布局)。
  • std::string * alias = &amp;s2;代替std::ptrdiff_t offset = (char*)(&amp;s2) - (char*)(this);,然后get变成return *alias;
  • @passing_through 使用成员指针而不是对象指针。然后,它不需要重置。

标签: c++ class language-lawyer offset undefined-behavior


【解决方案1】:

您提出的解决方案包含多个与指针算术相关的未定义行为实例。

第一个(char*)(&amp;s2) - (char*)(this) 是未定义的行为。此表达式由expr.add#5 管理。由于指针不是 nullptr 并且它们不指向同一数组中的元素,因此行为未定义。

第二个((char*)(this) + offset) 是未定义的行为。这次适用的段落是expr.add#4。由于(char*)(this) 不是数组元素,offset 的唯一合法值是 0。任何其他值都是未定义行为。

但是 C++ 已经提供了解决您所描述的问题所必需的工具:pointer to data member。这些指针指向 type 的成员,而不是 instance 的成员。它可以与指向实例的指针(在本例中为this 指针)结合以获得普通对象指针。

这是您的示例修改为使用指向数据成员 (https://godbolt.org/z/161vT158q) 的指针:

#include <cstddef>
#include <iostream>
#include <string>

class Test {
    std::string s1{"s1"}, s2{"s2"};

    // A pointer to an `std::string` member of the type `Test`
    using t_member_pointer = std::string Test::*;

    // Points to `Test::s2`
    t_member_pointer s_ptr = &Test::s2;

public:
    std::string& get() { 
        // Combine the data member pointer with an instance to get an object
        return (this->*s_ptr);
    }
};

int main() {
    Test test1;
    Test test2 = test1;
    std::cout << test2.get(); // note the copy
}

注意s_ptr 指向Test::s2 而不是this-&gt;s2。数据成员指针的值独立于任何实例,它与该类型的任何实例兼容。因此,在复制或移动期间不需要更正它,如果只是在实例之间按值复制,它将按预期运行。

【讨论】:

  • "因为 (char*)(this) 不是数组的元素" - 好吧,从技术上讲,任何变量/对象实例都可以被视为一个包含 1 个元素的数组,其中指针算术有关。
  • 因为 (char*)(this) 不是数组的元素,所以 offset 的唯一合法值是 0 因为 (char*)(this) 不指向char 类型的对象,即使 0 也不是合法的偏移量,参见 [expr.add]/6
【解决方案2】:

不,两个指针的区别只对来自同一个数组的指针有效:

只有指向同一数组元素的指针(包括数组末尾之后的指针)才能相互减去。

https://en.cppreference.com/w/cpp/types/ptrdiff_t

这不适用于类的不同成员。

【讨论】:

  • 请注意,这不是ptrdiff_t 的属性,它是关于指针运算的一般事实。
  • @FrançoisAndrieux 那么,有没有通过标准 C++ 实现这个想法的合法方法?
  • @passing_through 不,将您的成员放在同一个数组中或创建一个单独的指向相关成员的指针数组。考虑一个constexpr 成员指针数组。
  • 有一些宏巫术offsetof 虽然我不明白它是否或如何用于这种情况
  • @largest_prime_is_463035818 据我了解,offsetof 的用途是有限的。我不相信你可以用它来找回会员。编辑:它会给你正确的整数添加到this,但随后尝试执行该指针算术应该是UB,所以知道偏移量通常没有用。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2011-12-11
  • 2021-08-03
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2011-04-10
相关资源
最近更新 更多