关于共用体的内存详解

转载重点:共用体的所有成员占用同一段内存,修改一个成员会影响其余所有成员。图形分析影响过程

补充共用体所占内存大小为 成员变量中最大的所占内存大小,结构体所占内训大小为结构体中所有变量所占额内存大小。

通过前面的讲解,我们知道结构体(Struct)是一种构造类型或复杂类型,它可以包含多个类型不同的成员。在C语言中,还有另外一种和结构体非常类似的语法,叫做共用体(Union),它的定义格式为:

 
  1. union 共用体名{

  2.     成员列表

  3. };

共用体有时也被称为联合或者联合体,这也是 Union 这个单词的本意。

结构体和共用体的区别在于:结构体的各个成员会占用不同的内存,互相之间没有影响;而共用体的所有成员占用同一段内存,修改一个成员会影响其余所有成员。

结构体占用的内存大于等于所有成员占用的内存的总和(成员之间可能会存在缝隙),共用体占用的内存等于最长的成员占用的内存。共用体使用了内存覆盖技术,同一时刻只能保存一个成员的值,如果对新的成员赋值,就会把原来成员的值覆盖掉。

共用体也是一种自定义类型,可以通过它来创建变量,例如:

 
  1. union data{

  2. int n;

  3. char ch;

  4. double f;

  5. };

  6. union data a, b, c;

上面是先定义共用体,再创建变量,也可以在定义共用体的同时创建变量:

 
  1. union data{

  2. int n;

  3. char ch;

  4. double f;

  5. } a, b, c;

如果不再定义新的变量,也可以将共用体的名字省略:

 
  1. union{

  2. int n;

  3. char ch;

  4. double f;

  5. } a, b, c;

共用体 data 中,成员 f 占用的内存最多,为 8 个字节,所以 data 类型的变量(也就是 a、b、c)也占用 8 个字节的内存,请看下面的演示:

 
  1. #include <stdio.h>

  2. union data{

  3. int n;

  4. char ch;

  5. short m;

  6. };

  7. int main(){

  8. union data a;

  9. printf("%d, %d\n", sizeof(a), sizeof(union data) );

  10. a.n = 0x40;

  11. printf("%X, %c, %hX\n", a.n, a.ch, a.m);

  12. a.ch = '9';

  13. printf("%X, %c, %hX\n", a.n, a.ch, a.m);

  14. a.m = 0x2059;

  15. printf("%X, %c, %hX\n", a.n, a.ch, a.m);

  16. a.n = 0x3E25AD54;

  17. printf("%X, %c, %hX\n", a.n, a.ch, a.m);

  18. return 0;

  19. }

运行结果:

 
  1. 4, 4

  2. 40, @, 40

  3. 39, 9, 39

  4. 2059, Y, 2059

  5. 3E25AD54, T, AD54

这段代码不但验证了共用体的长度,还说明共用体成员之间会相互影响,修改一个成员的值会影响其他成员。

要想理解上面的输出结果,弄清成员之间究竟是如何相互影响的,就得了解各个成员在内存中的分布。以上面的 data 为例,各个成员在内存中的分布如下:

成员 n、ch、m 在内存中“对齐”到一头,对 ch 赋值修改的是前一个字节,对 m 赋值修改的是前两个字节,对 n 赋值修改的是全部字节。也就是说,ch、m 会影响到 n 的一部分数据,而 n 会影响到 ch、m 的全部数据。

上图是在绝大多数 PC 机上的内存分布情况,如果是 51 单片机,情况就会有所不同:

为什么不同的机器会有不同的分布情况呢?这跟机器的存储模式有关,我们将在VIP教程《大端小端以及判别方式》一节中展开探讨。

共用体的应用

共用体在一般的编程中应用较少,在单片机中应用较多。对于 PC 机,经常使用到的一个实例是: 现有一张关于学生信息和教师信息的表格。学生信息包括姓名、编号、性别、职业、分数,教师的信息包括姓名、编号、性别、职业、教学科目。请看下面的表格:

Name

Num

Sex

Profession

Score / Course

HanXiaoXiao

501

f

s

89.5

YanWeiMin

1011

m

t

math

LiuZhenTao

109

f

t

English

ZhaoFeiYan

982

m

s

95.0


f 和 m 分别表示女性和男性,s 表示学生,t 表示教师。可以看出,学生和教师所包含的数据是不同的。现在要求把这些信息放在同一个表格中,并设计程序输入人员信息然后输出。

如果把每个人的信息都看作一个结构体变量的话,那么教师和学生的前 4 个成员变量是一样的,第 5 个成员变量可能是 score 或者 course。当第 4 个成员变量的值是 s 的时候,第 5 个成员变量就是 score;当第 4 个成员变量的值是 t 的时候,第 5 个成员变量就是 course。

经过上面的分析,我们可以设计一个包含共用体的结构体,请看下面的代码:

 
  1. #include <stdio.h>

  2. #include <stdlib.h>

  3. #define TOTAL 4 //人员总数

  4. struct{

  5. char name[20];

  6. int num;

  7. char sex;

  8. char profession;

  9. union{

  10. float score;

  11. char course[20];

  12. } sc;

  13. } bodys[TOTAL];

  14. int main()

  15. {

  16. int i;

  17. //输入人员信息

  18. for(i=0; i<TOTAL; i++)

  19. {

  20. printf("Input info: ");

  21. scanf("%s %d %c %c", bodys[i].name, &(bodys[i].num), &(bodys[i].sex) &(bodys[i].profession));

  22. if(bodys[i].profession == 's')

  23. { //如果是学生

  24. scanf("%f", &bodys[i].sc.score);

  25. }

  26. else

  27. { //如果是老师

  28. scanf("%s", bodys[i].sc.course);

  29. }

  30. fflush(stdin);

  31. }

  32. //输出人员信息

  33. printf("\nName\t\tNum\tSex\tProfession\tScore / Course\n");

  34. for(i=0; i<TOTAL; i++)

  35. {

  36. if(bodys[i].profession == 's')

  37. {

  38. //如果是学生

  39. printf("%s\t%d\t%c\t%c\t\t%f\n", bodys[i].name, bodys[i].num, bodys[i].sex, bodys[i].profession, bodys[i].sc.score);

  40. }

  41. else

  42. {

  43. //如果是老师

  44. printf("%s\t%d\t%c\t%c\t\t%s\n", bodys[i].name, bodys[i].num, bodys[i].sex, bodys[i].profession, bodys[i].sc.course);

  45. }

  46. }

  47. return 0;

  48. }

运行结果:

 
  1. Input info: HanXiaoXiao 501 f s 89.5↙

  2. Input info: YanWeiMin 1011 m t math↙

  3. Input info: LiuZhenTao 109 f t English↙

  4. Input info: ZhaoFeiYan 982 m s 95.0↙

  5. Name Num Sex Profession Score / Course

  6. HanXiaoXiao 501 f s 89.500000

  7. YanWeiMin 1011 m t math

  8. LiuZhenTao 109 f t English

  9. ZhaoFeiYan 982 m s 95.000000

  • 1
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值