超凡天赐 发表于 2017-4-30 10:39:23

关于结构体套联合体的问题

先贴一些代码
#include <stdio.h>
struct text
{
    int a;
    char b;
    float c;
    union u
    {
      char u1;
      int u2;
    } ua;
};
int main()
{
    printf("%lu\n",sizeof(struct text));
    return 0;
}
我们会发现它的运行结果
20
Program ended with exit code: 0
那么问题来了,较之前的原则:
1.前面地址必须是后面地址的整数倍,不是就补齐;
2.整个结构体的地址必须是最大字节的整数倍;
但是在上述代码中,我们发现测出的结果竟然是20,整个结构体并不是最大字节的整数倍。ua 所占的字节明显是8,而20也明显不是8的整数倍。
那么我就猜想了,如果发生了结构体套用联合体的问题,那么联合体默认的最大字节数并不是其本身的字节数,而是联合体中最大字节数。如果这样的话,就合理解释了上面的情况。可是我有撸了一串代码,发现我的观念又错了。{:10_247:}
#include <stdio.h>
struct text
{
    struct text *p;
    char c;
    int a;
    union u
    {
      int b;
      char a;
    }ua;
};
int main()
{
    printf("%lu\n",sizeof(struct text));
    return 0;
}
运行结果
40
Program ended with exit code: 0
这到底是为什么?网上找不到答案了,来求助大神们了。sorry,又来and你们了。{:10_297:} @人造人 @四十二 @~风介~ @qq1242009750

超凡天赐 发表于 2017-4-30 10:40:45

@小甲鱼

人造人 发表于 2017-4-30 11:32:34

自己分析
哪里看不懂,问我^_^

#include <stdio.h>
#include <stddef.h>

struct text
{
        int a;
        char b;
        float c;
        union u
        {
                char u1;
                int u2;
        } ua;
};

int main(void)
{
        printf("struct text size: %d\n", sizeof(struct text));

        putchar('\n');

        printf("a offset : %d\n", offsetof(struct text, a));
        printf("b offset : %d\n", offsetof(struct text, b));
        printf("c offset : %d\n", offsetof(struct text, c));
        printf("u1 offset : %d\n", offsetof(struct text, ua.u1));
        printf("u2 offset : %d\n", offsetof(struct text, ua.u2));

        putchar('\n');

        struct text tmp;
        printf("ua size: %d\n", sizeof(tmp.ua));

        return 0;
}


struct text size: 20

a offset : 0
b offset : 4
c offset : 8
u1 offset : 12
u2 offset : 12

ua size: 8
请按任意键继续. . .

人造人 发表于 2017-4-30 11:37:27

#include <stdio.h>

struct text
{
        struct text *p;
        char c;
        int a;
        union u
        {
                int b;
                char a;
        }ua;
};

int main(void)
{
        printf("%lu\n", sizeof(struct text));
        return 0;
}


32
请按任意键继续. . .

超凡天赐 发表于 2017-4-30 12:54:55

人造人 发表于 2017-4-30 11:32
自己分析
哪里看不懂,问我^_^

1.前面地址必须是后面地址的整数倍,不是就补齐;
2.整个结构体的地址必须是最大字节的整数倍;
我的问题是,联合体的字节数8字节,而结构体的字节数是20字节,在这个程序中,结构体应该是8的倍数呀??

超凡天赐 发表于 2017-4-30 12:55:29

本帖最后由 超凡天赐 于 2017-4-30 12:58 编辑

超凡天赐 发表于 2017-4-30 12:54
1.前面地址必须是后面地址的整数倍,不是就补齐;
2.整个结构体的地址必须是最大字节的整数倍;
我的问 ...
{:10_247:}

超凡天赐 发表于 2017-4-30 12:58:17

人造人 发表于 2017-4-30 11:37


在我的电脑中是40,你的编译器是32位的?

人造人 发表于 2017-4-30 13:14:12

超凡天赐 发表于 2017-4-30 12:54
1.前面地址必须是后面地址的整数倍,不是就补齐;
2.整个结构体的地址必须是最大字节的整数倍;
我的问 ...

貌似对齐的是基本类型
联合体好像不算,数组也不算

好像只有
char
short
int
long
float
double

似乎只有这些

还有,这个只做了解就好,在现实编程中很少会用到

人造人 发表于 2017-4-30 13:16:33

超凡天赐 发表于 2017-4-30 12:58
在我的电脑中是40,你的编译器是32位的?

嗯,改成64位后,输出
40
请按任意键继续. . .

超凡天赐 发表于 2017-4-30 13:41:40

人造人 发表于 2017-4-30 13:14
貌似对齐的是基本类型
联合体好像不算,数组也不算



还有指针类型{:10_334:}

人造人 发表于 2017-4-30 13:45:53

超凡天赐 发表于 2017-4-30 13:41
还有指针类型

^_^

SmallDream 发表于 2017-5-14 22:51:42

前辈的对话与对细节的要求让我感到惭愧,同时勾起了对这些知识的回忆,谢谢前辈
页: [1]
查看完整版本: 关于结构体套联合体的问题