鱼C论坛

 找回密码
 立即注册
查看: 2442|回复: 11

[已解决]关于结构体套联合体的问题

[复制链接]
发表于 2017-4-30 10:39:23 | 显示全部楼层 |阅读模式

马上注册,结交更多好友,享用更多功能^_^

您需要 登录 才可以下载或查看,没有账号?立即注册

x
先贴一些代码
  1. #include <stdio.h>
  2. struct text
  3. {
  4.     int a;
  5.     char b;
  6.     float c;
  7.     union u
  8.     {
  9.         char u1[5];
  10.         int u2[2];
  11.     } ua;
  12. };
  13. int main()
  14. {
  15.     printf("%lu\n",sizeof(struct text));
  16.     return 0;
  17. }
复制代码

我们会发现它的运行结果
  1. 20
  2. Program ended with exit code: 0
复制代码

那么问题来了,较之前的原则:
1.前面地址必须是后面地址的整数倍,不是就补齐;
2.整个结构体的地址必须是最大字节的整数倍;
但是在上述代码中,我们发现测出的结果竟然是20,整个结构体并不是最大字节的整数倍。ua 所占的字节明显是8,而20也明显不是8的整数倍。
那么我就猜想了,如果发生了结构体套用联合体的问题,那么联合体默认的最大字节数并不是其本身的字节数,而是联合体中最大字节数。如果这样的话,就合理解释了上面的情况。可是我有撸了一串代码,发现我的观念又错了。
  1. #include <stdio.h>
  2. struct text
  3. {
  4.     struct text *p;
  5.     char c;
  6.     int a[2];
  7.     union u
  8.     {
  9.         int b[4];
  10.         char a[3];
  11.     }ua;
  12. };
  13. int main()
  14. {
  15.     printf("%lu\n",sizeof(struct text));
  16.     return 0;
  17. }
复制代码

运行结果
  1. 40
  2. Program ended with exit code: 0
复制代码

这到底是为什么?网上找不到答案了,来求助大神们了。sorry,又来and你们了。 @人造人 @四十二 @~风介~ @qq1242009750
最佳答案
2017-4-30 13:14:12
超凡天赐 发表于 2017-4-30 12:54
1.前面地址必须是后面地址的整数倍,不是就补齐;
2.整个结构体的地址必须是最大字节的整数倍;
我的问 ...

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

好像只有
char
short
int
long
float
double

似乎只有这些

还有,这个只做了解就好,在现实编程中很少会用到
小甲鱼最新课程 -> https://ilovefishc.com
回复

使用道具 举报

 楼主| 发表于 2017-4-30 10:40:45 | 显示全部楼层
@小甲鱼
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2017-4-30 11:32:34 | 显示全部楼层

回帖奖励 +10 鱼币

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

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

  3. struct text
  4. {
  5.         int a;
  6.         char b;
  7.         float c;
  8.         union u
  9.         {
  10.                 char u1[5];
  11.                 int u2[2];
  12.         } ua;
  13. };

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

  17.         putchar('\n');

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

  23.         putchar('\n');

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

  26.         return 0;
  27. }
复制代码

  1. struct text size: 20

  2. a offset : 0
  3. b offset : 4
  4. c offset : 8
  5. u1 offset : 12
  6. u2 offset : 12

  7. ua size: 8
  8. 请按任意键继续. . .
复制代码

评分

参与人数 1荣誉 +5 鱼币 +5 贡献 +2 收起 理由
超凡天赐 + 5 + 5 + 2 支持楼主!

查看全部评分

小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2017-4-30 11:37:27 | 显示全部楼层
  1. #include <stdio.h>

  2. struct text
  3. {
  4.         struct text *p;
  5.         char c;
  6.         int a[2];
  7.         union u
  8.         {
  9.                 int b[4];
  10.                 char a[3];
  11.         }ua;
  12. };

  13. int main(void)
  14. {
  15.         printf("%lu\n", sizeof(struct text));
  16.         return 0;
  17. }
复制代码

  1. 32
  2. 请按任意键继续. . .
复制代码
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2017-4-30 12:54:55 | 显示全部楼层
人造人 发表于 2017-4-30 11:32
自己分析
哪里看不懂,问我^_^

1.前面地址必须是后面地址的整数倍,不是就补齐;
2.整个结构体的地址必须是最大字节的整数倍;
我的问题是,联合体的字节数8字节,而结构体的字节数是20字节,在这个程序中,结构体应该是8的倍数呀??
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2017-4-30 12:55:29 | 显示全部楼层
本帖最后由 超凡天赐 于 2017-4-30 12:58 编辑
超凡天赐 发表于 2017-4-30 12:54
1.前面地址必须是后面地址的整数倍,不是就补齐;
2.整个结构体的地址必须是最大字节的整数倍;
我的问 ...

小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2017-4-30 12:58:17 | 显示全部楼层

在我的电脑中是40,你的编译器是32位的?
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2017-4-30 13:14:12 | 显示全部楼层    本楼为最佳答案   
超凡天赐 发表于 2017-4-30 12:54
1.前面地址必须是后面地址的整数倍,不是就补齐;
2.整个结构体的地址必须是最大字节的整数倍;
我的问 ...

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

好像只有
char
short
int
long
float
double

似乎只有这些

还有,这个只做了解就好,在现实编程中很少会用到
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2017-4-30 13:16:33 | 显示全部楼层
超凡天赐 发表于 2017-4-30 12:58
在我的电脑中是40,你的编译器是32位的?

嗯,改成64位后,输出
  1. 40
  2. 请按任意键继续. . .
复制代码
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2017-4-30 13:41:40 | 显示全部楼层
人造人 发表于 2017-4-30 13:14
貌似对齐的是基本类型
联合体好像不算,数组也不算

还有指针类型
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2017-4-30 13:45:53 | 显示全部楼层
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2017-5-14 22:51:42 | 显示全部楼层

回帖奖励 +10 鱼币

前辈的对话与对细节的要求让我感到惭愧,同时勾起了对这些知识的回忆,谢谢前辈
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Archiver|鱼C工作室 ( 粤ICP备18085999号-1 | 粤公网安备 44051102000585号)

GMT+8, 2025-7-1 05:56

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表