这篇文章是一篇译文,跟上一篇文章相呼应的,原文在这里

对于结构体和空类大小是1这个问题,首先这是一个C++问题,在C语言下空结构体大小为0(当然这是编译器相关的)。这里的空类和空结构体是指类或结构体中没有任何成员。

在C++下,空类和空结构体的大小是1(编译器相关),这是为什么呢?为什么不是0?

这是因为,C++标准中规定,“no object shall have the same address in memory as any other variable” ,就是任何不同的对象不能拥有相同的内存地址。 如果空类大小为0,若我们声明一个这个类的对象数组,那么数组中的每个对象都拥有了相同的地址,这显然是违背标准的。

但是,也许你还有一个疑问,为什么C++标准中会有这么无聊的规定呢?

当然,这样规定显然是有原因的。我们假设C++中有一个类型T,我们声明一个类型T的数组,然后再声明一个T类型的指针指向数组中间某个元素,则我们将指针减去1,应该得到数组的另一个索引。如下代码:

1
2
3
4
5
T array[5];
 
int diff = &array[3] - &array[2];
 
 // diff = 1

上面的代码是一种指针运算,将两个指针相减,编译器作出如下面式子所示的动作:

diff = ((char *)&array[3] - (char *)&array[2]) / sizeof T;

式子应该不难懂把,很明显的一点就是这个式子的计算依赖于sizeof T。虽然上面只是一个例子,但是基本上所有的指针运算都依赖于sizeof T。

好,下面我们来看,如果允许不同的对象有相同的地址将会引发什么样的问题,看下面的例子:

1
2
3
4
 &array[3] - &array[2] = &array[3] - &array[1]
                       = &array[3] - &array[1]
                       = &array[3] - &array[0]
                       = 0

我们可以看到,在这个例子中,如果每个对象都拥有相同地址,我们将没有办法通过指针运算来区分不同的对象。还有一个更严重的问题,就是如果 sizeof T是0,就会导致编译器产生一个除0的操作,引发不可控的错误。

基于这个原因,如果允许结构体或者类的大小为0,编译器就需要实现一些复杂的代码来处理这些异常的指针运算。

所以,C++标准规定不同的对象不能拥有相同的地址。那么怎样才能保证这个条件被满足呢?最简单的方法莫过于不允许任何类型的大小为0。所以编译器为每个空类或者空结构体都增加了一个虚设的字节(有的编译器可能加的更多),这样这些空类和空结构的大小就不会是0,就可以保证他们的对象拥有彼此独立的地址。

anyShare分享到:
Tagged with:
 

9 Responses to “为什么C++中空类和空结构体大小为1?”

  1. tanghao says:

    很给力

  2. s007 says:

    "如果每个对象都拥有相同地址,我们将没有办法通过指针运算来区分不同的对象。如果允许结构体或者类的大小为0,编译器就需要实现一些复杂的代码来处理这些异常的指针运算。",

    C语言编译器来说不搞定了吗~~感觉写编译器的大神不会被这个问题难倒~~

    这是实现编译器的人/或者设计语言的人的一种设计取向,避免复杂,便于实现,还是说另有别的原因呢~?

    • sponge says:

      @s007 实现毫无疑问是没有问题的。c++和c语言不同之一就是c++是面向对象的,很强调对象,所以每个对象应该都需要一个实体,可以避免复杂,又可以更好的符合软件工程。我自己想的,不一定正确,呵呵

  3. JonsenElizee says:

    Mark: C++ class size

  4. AttaCKovER says:

    ljn? 呵呵

    不过我看到gcc在语法树标记size时,unknown type全部标记为1。当时就想可能是因为标准有规定,不过没想到是因为说不同的对象必须要有不同的地址。如果要解释的话,我倒觉这应该是一个理念,很自然。

  5. tangxuan says:

    我是来留言的,顺便作广告。

  6. remychan says:

    got it
    good job.
    continue

  7. loopcm says:

    表示只是0而已

  8. [...] 为什么C++中空类和空结构体大小为1? > 这是因为,C++标准中规定,“no object shall have the same address in memory as any other variable” ,就是任何不同的对象不能拥有相同的内存地址。 如果空类大小为0,若我们声明一个这个类的对象数组,那么数组中的每个对象都拥有了相同的地址,这显然是违背标准的。 > 基本上所有的指针运算都依赖于sizeof T。 [...]

Leave a Reply

Note: Commenter is allowed to use '@User+blank' to automatically notify your reply to other commenter. e.g, if ABC is one of commenter of this post, then write '@ABC '(exclude ') will automatically send your comment to ABC. Using '@all ' to notify all previous commenters. Be sure that the value of User should exactly match with commenter's name (case sensitive).