班级规模 - C++

发布于 2025-01-04 15:31:43 字数 405 浏览 1 评论 0原文

我有以下关于对象大小的代码:

class A 
{ 
public: 
    int _i; 
    virtual int getI () = 0; 
    int setI (int i); 
}; 
class B : public A 
{ 
public: 
    int getI (); 
    virtual int setI (int i); 
}; 

class C : public B 
{ 
public: 
    int _i; 
    int getI (); 
    int setI (int i); 
}; 

int main () 
{ 
    B b; 
    C c; 
} 

Why the size of Cc; 是12?尺寸计算中包括哪些部分?

I have the following code about object sizes:

class A 
{ 
public: 
    int _i; 
    virtual int getI () = 0; 
    int setI (int i); 
}; 
class B : public A 
{ 
public: 
    int getI (); 
    virtual int setI (int i); 
}; 

class C : public B 
{ 
public: 
    int _i; 
    int getI (); 
    int setI (int i); 
}; 

int main () 
{ 
    B b; 
    C c; 
} 

Why the size of
C c;
is 12? What parts included in the size calculation?

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(5

酒浓于脸红 2025-01-11 15:31:43

sizeof(int A::_i) + sizeof(int C::_i) + sizeof(pointer to virtual table)

所有这些部分的大小取决于实现,在您的情况下,每个部分都有尺寸 4。

sizeof(int A::_i) + sizeof(int C::_i) + sizeof(pointer to virtual table)

The size of all these parts is implementation-dependent, in your case, each one has size 4.

最后的乘客 2025-01-11 15:31:43

几乎任何类的大小都取决于实现,但我会
猜测你在 32 位机器上,并且 C 类包含 4 个字节
vptr 和两个四字节 intA::_iC::_i)。

The size of just about any class is implementation dependent, but I'll
guess that you're on a 32 bit machine, and class C contains a 4 byte
vptr and two four byte int (A::_i and C::_i).

风月客 2025-01-11 15:31:43

对象的大小将是特定于平台的。例如,在 64 位平台上,我希望大小为 24 字节。物体的大小由什么构成有点棘手。它由各种组件组成:

  • 大小的明显组件是数据成员:在您的情况下,您在层次结构中的某个位置有两个 int ,即这将贡献 2 * sizeof( int)
  • 众所周知的隐藏组件是指向“vtable”的指针,即指向某种数据结构的指针,该数据结构处理虚拟函数的调用方式。这通常会贡献一个指针大小。
  • 大多数人可能完全忽略了多重继承带来的隐藏指针,特别是当对象涉及虚拟基类时:为了使对象看起来像是特定层次结构中的一个,它将包含比“vtable”指针更多的内容。
  • 尽管被许多人忽视,但更广为人知的是“填充”,即用于确保数据成员与 CPU 首选地址对齐的隐藏字节。一般来说,如果存在与此大小相对应的基本类型,则首选对齐方式大致为给定系统上的类型大小,最多为给定系统上的高速缓存行的大小。也就是说,通常最大对齐为 16 字节。
  • 最后,我可以想到一个有趣的空类:由于子对象需要具有不同的地址,因此每个空类将被视为至少一个字节大,除非空类是基类,在这种情况下它可以共享与其他基地相同的地址。

大多数隐藏的内容并不适用,但对于您的类,您有 2 * sizeof(int) + sizeof(T*) 用于访问虚拟函数表的合适类型。

The size of the object will be platform specific. For example on a 64 bit platform I would expect the size to be 24 bytes. What constitutes the size of an object is a bit tricky. It is made up from variety of components:

  • The obvious components of the size are the data members: in your case you got two int somewhere in the hierarchy, i.e. this will contribute 2 * sizeof(int).
  • The well-known hidden component is a pointer to the "vtable", i.e. a pointer to some sort of data structure which deals with the ways virtual functions are called. This will generally contribute a pointer size.
  • Probably entirely ignored by most are the hidden pointers coming with the multiple inheritance especially when the object involves a virtual base class: to make the object look as if it is one in specific hierarchy it will include more than on "vtable" pointer.
  • More known although ignored by many is "padding" i.e. hidden bytes used to assure that data members are aligned to addresses preferred by the CPU. In general, the preferred alignment is roughly the size of the type up to the size of a cache line on the given system if there are corresponding fundamental types of this size. That is, typically the maximum alignment is 16 bytes.
  • Finally, I can think of a fun one with empty classes: since subobjects need to have distinct addresses, each empty class will by treated as if it is at least one byte big unless the empty class is a base class in which case it may share the same address as other bases.

Most of the hidden stuff doesn't apply but for you class you have 2 * sizeof(int) + sizeof(T*) for a suitable type used to access the virtual function table.

生寂 2025-01-11 15:31:43

C c; 的大小实际上是实现定义的。如果您有任何实际依赖于大小的代码,则该代码是严重错误的,并且在切换编译器时可能会中断。

现在回答您的实际问题:您可能认为 C 具有 C 中包含的 int 的大小,或者您可能认为它具有来自 Cint 的大小加上来自 A 的大小。

这两种猜测都是错误的,原因有两个:

  • 您的结构中可能存在所谓的填充。有时类型需要与某些边界对齐。为了强制这些对齐,编译器在字段之间引入了一些“浪费”的空间区域,以将它们保留在这些位置。您永远不能依赖此填充的数量。

  • 还有一点,我主要认为你的老师想向你展示这一点:如果你编写c.getI(),计算机必须知道要调用哪个方法,即是否调用来自 ABC 之一。这些信息需要存储在某个地方。此信息的存储会为您的结构添加一些额外的大小,但您永远不能依赖会添加多少。有些人可能会试图告诉您这只是使用一个指针来存储的,但这是不正确的。编译器可以使用尽可能多的空间来存储这些信息。大多数编译器出于效率原因只使用一个指针,但如果您依赖这一点,对于以不同方式处理此问题的编译器来说,您的代码将会是错误的。

如果您想了解更多信息,只需谷歌搜索“虚拟方法调用”,您就可以找到该用例的常见实现的一些示例。

The size of C c; is actually implementation defined. If you have any code that actually relies on the size, this code is horribly wrong and may break whenever the compiler is switched.

Now to your actuall questions: You probably either thought C would have the size of the int contained in C or alternatively you would have thought it to have the size of the int from C plus the one from A.

Both these guesses are wrong for two reasons:

  • There might be so called padding in your structures. Sometimes types need to be aligned to certain boundaries. In order to enforce these alignments, the compiler introduces some area of "wasted" space between your fields to keep them at these places. You can never rely on the amount of this padding.

  • There is another point, and I mainly think your teacher wanted to show you this: If you write c.getI() the computer has to know which method to call, i.e. whether to call the one from A, B or C. This information needs to be stored somewhere. The storage for this information adds some additional size to your struct, but you can never rely on how much will be added. Some people might try to tell you that this is only stored by using one pointer, but this is not correct. The compiler is allowed to use as much space to store this information as it likes. Most compiler only do use one pointer for efficiency reasons, but if you rely on this, your code will be wrong for compilers which do handle this differently.

If you want to know more, just google for "virtual method invocation", and you can find some examples for common implementations for this use-case.

凉薄对峙 2025-01-11 15:31:43

我认为你使用 32 位编译器。
这是C类的内存布局:

class C size(12):
    +---
    | +--- (base class B)
    | | +--- (base class A)
 0  | | | {vfptr}
 4  | | | _i
    | | +---
    | +---
 8  | _i
    +---

C::$vftable@:
    | &C_meta
    |  0
 0  | &C::getI
 1  | &C::setI

根据它,你可以看到哪些部分影响类的大小。

您可以阅读有关虚拟函数、虚拟继承、虚拟表的更多信息,以了解它们的组织方式以及对类大小的影响。

如果您使用 MVSC,则可以在编译时使用 -d1reportAllClassLayout 选项来查看类的布局。

I think you use 32 bit compiler.
This is the memory layout of class C:

class C size(12):
    +---
    | +--- (base class B)
    | | +--- (base class A)
 0  | | | {vfptr}
 4  | | | _i
    | | +---
    | +---
 8  | _i
    +---

C::$vftable@:
    | &C_meta
    |  0
 0  | &C::getI
 1  | &C::setI

Based on it, you can see which parts conduct to size of class.

You can read more about virtual function, virtual inheritance, virtual table to figure out the way that they are organized and contributed to class size.

If you use MVSC, you can use -d1reportAllClassLayout option when you compile to see the layout of your class.

~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文