结构具有联合时的意外大小

时间:2018-02-14 07:38:43

标签: c struct sizeof structure-packing

我正在尝试验证结构的大小。由于某些原因,它给我的大小为18而不是预期的14个字节(联合应该最大为8 + 2 + 2 = 12个字节)。有人能帮助我吗?

typedef struct __attribute__((__packed__)) def
{
        union {
                double x;   /* 8 bytes */
                char *y;    /* 8 bytes as for 64-bit system */
                struct {
                        struct def *array; /* 8 bytes */
                        unsigned short a;  /* 2 bytes */
                        unsigned short b;  /* 2 bytes */
                } z;
        } w;
        unsigned short v;  /* 2 bytes */
} DEF, *DEFP;

int main(int argc, char **argv) {
        printf("size of DEF = %lu\n", sizeof(DEF));
        printf("size of unsigned short = %lu\n", sizeof(unsigned short));
        printf("size of struct def * = %lu\n", sizeof(struct def *));
        printf("size of char * = %lu\n", sizeof(char *));
        printf("size of double = %lu\n", sizeof(double));
}

这是我运行时显示的内容:

$ gcc test.c && ./a.out
size of DEF = 18
size of unsigned short = 2
size of struct def * = 8
size of char * = 8
size of double = 8

2 个答案:

答案 0 :(得分:3)

__attribute__((__packed__))仅指struct def。它不会将匿名struct打包在union内的匿名struct def内。

所以很可能那两个成员

                   unsigned short a;  /* 2 bytes */
                   unsigned short b;  /* 2 bytes */

“使用”4但是2个字节。

与您的问题无关:C要求使用size_t 长度修饰符打印z

 printf("size of DEF = %zu\n", sizeof (DEF));

答案 1 :(得分:2)

写作:

struct foo {
    struct bar {
        ...
    } x;
};

与写作没有什么不同:

struct bar { ... };
struct foo {
    struct bar x;
};

为什么重要?这很重要,因为在第二个示例中,应该明确为什么应用于foo的属性不会自动应用于bar。这意味着您的外部结构将被打包,但这不会改变组件的含义。

你所拥有的内部结构将不会被打包,并且很明显,如果它遵循通常的对齐规则,大多数体系结构遵循其大小将是其最严格对齐的成员的多个,它将​​是指针。因此,union中结构的大小将为16,而不是您假设的12。这意味着联盟的大小也将是16。