1、struct和union结构体和联合体的区别(转) 共用体 构造数据类型,也叫联合体 用途:使几个不同类型的变量共占一段内存(相互覆盖) 结构体是一种构造数据类型 用途:把不同类型的数据组合成一个整体-------自定义数据类型 --------------------------------------------------------------- 结构体变量所占内存长度是各成员占的内存长度的总和。 共同体变量所占内存长度是各最长的成员占的内存长度。 共同体每次只能存放哪个的一种!! 共同体变量中起作用的成员是尊后
2、一次存放的成员, 在存入新的成员后原有的成员失去了作用! --------------------------------------------------------------- Structure 与 Union主要有以下区别: 1. struct和union都是由多个不同的数据类型成员组成, 但在任何同一时刻, union中只存放了一个被选中的成员, 而struct的所有成员都存在。在struct中,各成员都占有自己的内存空间,它们是同时存在的。一个struct变量的总长度等于所有成员长度之 和。在Union中,所有成员不能同时占用它的内存空间,它们不
3、能同时存在。Union变量的长度等于最长的成员的长度。
2. 对于union的不同成员赋值, 将会对其它成员重写, 原来成员的值就不存在了, 而对于struct的不同成员赋值是互不影响的。
举一个例子:
例:
#include
4、 /*在联合中定义一个结构*/ char first; char second; }half; }number; number.i=0x4241; /*联合成员赋值*/ printf("%c%c\n", number.half.first, number.half.second); number.half.first='a'; /*联合中结构成员赋值*/ number.half.second='b'; printf
5、"%x\n", number.i); system("pause"); } 输出结果为: AB 6261 分析: union的成员是共用内存的 union{ int i; struct{ char first; char second; }half; }number; number.i=0x4241; 在这里i 和 half结构是共用内存 number.i=0x4241给i赋值后,内存中以二进制存储0100 0010 0100 0001 按顺序对应到结构中 halt.firs
6、t=01000010 转换成10进制就是66(字母A的asc码)
halt.second=01000001 转换成10进制是65 (字母B的asc码)
所以输出后就是 AB
下面同理了
==========================================================================
第一题:
#include
7、"%d",a.i); } 答案:266 第二题: main() { union{ /*定义一个联合*/ int i; struct{ /*在联合中定义一个结构*/ char first; char second; }half; }number; number.i=0x4241; /*联合成员赋值*/ printf("%c%c\n", number.half.first, mumber.half.second); number.half.first='a'; /*联合中结构成员赋值*/ number.half.second='b'
8、 printf("%x\n", number.i); getch(); } 答案: AB 6261 C语言中的联合体(UNION)的概念是,联合体中的多种数据类型共享同一个内存空间。就拿你举的例子来说: union { int i; char x[2]; }a; 在联合体a中定义了两种数据类型,字符数组x以及整形变量i.其中整形变量是16位的,数组大小为2的字符数组为8X2=16位。如此一来,编译器便会为 联合体a在内存中开辟一个16位的空间,这个空间里存储联合体的数据,但是这个空间只有16位,它既是整形变量的数据,也是字符数组的数据。如果你的程序
9、从字符数组的角度解析这个空间,那么它就是两个字符,如果你的程序从整型的角度解析这个空间,那么它就是一个整数。 以你的程序为例子,现在已经开辟了一个16位的空间,然后我们假定现在空间还没有被赋值,为: 00000000 00000000 那么在运行完代码 a.x[0] = 10; a.x[1] = 1; 之后,16位的空间变为: 00000110 00000001 然后程序运行 printf("%d",a.i); 就是把联合体a当成一个整数来解析,而不是字符串数组。那么这样一来,程序就把这16位变成了一个完整的整数: (00000001 0000011
10、0)二进制 = (266)十进制 注意,你可以看到程序在把16位弄成整数的时候把后面八位放在了前面,前面八位放在了后面。这个反序是计算机存储结构造成的,这个和联合体没有直接关系。如果感兴趣的话可以参考汇编语言。 就是这个道理。 第二个例子同样, union{ /*定义一个联合*/ int i; struct{ /*在联合中定义一个结构*/ char first; char second; }half; }number; 定义了联合体number,这个联合体有两种数据类型,整形i(16位),以及一个结构体(struct half)(2个char,16
11、位)。所以编译器为这个联合体开辟一个16位的空间: 00000000 00000000 然后赋值: number.i=0x4241; 这个时候,联合体以整形的身份出现,16位的空间将被整体认为是一个整数赋值。 注意(0x4241)(16进制) = (01000010 01000001)二进制。还记得刚才说的,计算机存储的时候是反着存的吗,先存低位,再存高位(参考汇编语言),因此16位地址被赋值位 01000001 01000010 然后 printf("%c%c\n", number.half.first, mumber.half.second); 实际上是
12、把16位空间以结构体half的角度解析,也就是两个char. 那么第一个:number.half.first = (01000001)二进制 = (65)十进制 = A(ASCII码) 同理number.half.second = B(ASCII码) 当然后头又给first和second赋值位"a"和"b",这样会把16位空间弄成是: 01100001 01100010 然后用 printf("%x\n", number.i); 就是把16位看成整数,记住高地位反过来 (01100010 01100001)二进制 = (0X6261)16进制 所以结果就是:0x6261. getch(); 最后记得按任意键结束程序。






