struct结构体是由一系列相同类型或不同类型的数据构成的数据集合。结构体也称结构。结构体的一般定义形式如下:
struct结构体名称
{
成员表列
}变量表列;
例如:
/*定义一个名称为 Person的结构体*
struct Person
{
char *pszName; /*Person的名称*/
Inti Sex; /*Person 的性别*/
}Boy;//定义一个名称为Boy的变量
union 联合由几个不同的变量存放在同一块区域内,也就是覆盖技术,几个变量互相覆盖。union 联合的一般定义形式如下:
union结构体名称
{
成员表列
}变量表列;
例如:
/*在同一位置存放1字和2字节*/
union WordToByte
{
unsigned char Tbyte[2]; /*Tbyte[0]是Tword 的低字节*/
unsigned short Tword; /*Tbyte[1]是Tword 的高字节*/
}Word;
从定义形式上看,“联合”和“结构体”非常形似,但二者在本质上却有很大差异。首先,在结构体中每个成员都有自己的内存,一个结构体变量的总长度等于各个成员长度的总和。而在联合中,各个成员共享一段内存空间,一个联合变量的长度等于各个成员中长度最大的那个成员的长度。联合体对数据的引用与结构体对数据的引用方式相同,都必须通过联合或结构体中的元素引用,而不能引用联合体变量。
例如,针对上述定义的联合体变量Word,下面的操作就是错误的:
printf("%d",Word); <== 这种用法是错误的
说明:联合体的共享不是指把各个成员同时装入一个联合体内,而是数据根据联合体的定义可以展现成不同的形式,同时向用户传递不同信息的过程。
联合体一般有下述几个特点:
●同一个联合体可用来存放几种不同类型的成员,但是在每一瞬间只能存放其中的一种,而不是同时存放几种。换句话说,每一瞬间只有一个成员起作用,其他的成员不起作用。
●共用体变量中起作用的成员是最后一次存放的成员,在存入一个新成员后,原有成员就失去作用
●共用体变量的地址和其各成员的地址都是同一地址。
●不能对共用体变量名赋值,也不能企图引用变量名来得到一个值,并且也不能在定义共用体变量时对它进行初始化。
●不能把共用体变量作为函数参数,也不能作为函数的返回值,但可使用指向共用体变量的指针。
●共用体类型可出现在结构体类型的定义中,也可定义共用体数组;反之,结构体也可出现在共用体类型的定义中,数组也可作为共用体的成员。
最后,我们来看一下联合体两种经典的应用场景:第一种是通过联合体判断操作系统或者CPU的特征;第二种是同样的数据采用不同的展现方式,会使获取的信息更清晰明了。
首先看第一种:获取操作系统或者 CPU 的特征。我们经常会碰到这样的问题:在网络传输中数据都是采用大端传输的,而我们不知道服务器CPU 采用的格式是大端的还是小端的,在编写服务器处理程序时有时会遇到数据解析错误的现象,采用联合体可方便地获取CPU的大小端特征。
union w
{
int a: //4 bytes
char b; //1 byte
}c;
c.a=1;
if(c.b==1)
{
printf("此CPU是小端CPU\n");
}
Else
{
printf("此CPU是大端CPU\n");
}
接着,我们将采用联合体实现数据的展示。同样以网络数据为例。在网络中数据都是以流的形式传输的,而当到达服务器时,数据流就会以不同的形式解析。典型的就是包头的解析。我们看一个链路层数据报头通过联合处理后数据的展示:
//网络数据,全部是十六进制数据,不方便报文的分析和查看
szData[]={
0x70,0x1a,0x04,0xae,0x2f,0x0f,0x40,0x16,0x9f,0xd6,0x62,0x8a,0x08,0x00,0x45,
0x00,0x00,0x2c,0x45,0x62,0x00,0x00,0x32,0x11,0xa4,0xa9,0xdb,0x8f,0x01,0x1a,
0xc0,0xa8,0x01,0x64,0xaa,0xd0,0x11,0x72,0x00,0x18,0x55,0x5b,0x4b,0x55,0x00,
0x01,0x04,0x000x00,0x08,0x00,0x00,0x00,0x03,0x00,0x00,0x00,0x09
};
//链路层MAC数据报头的定义
struct eth hdr
{
struct eth addr dest; //目标MAC地址
struct eth addr src; //源MAC地址
u16_ttype; //类型
}:
union ETH_HDR //eth头联合体
{
char*pszMacData:
eth hdr ethhdr;
}
ETH_HDR ethHDR:
memset(&thMsg, 0, sizeof(ETH_HDR));
ethHDR.pszMacData= szData; // szData代表网络上读取的数据
通过 ethHDR.ethhdr.type 可方便地查看 MAC 报文头中的数据类型。
请谨记
联合体虽然和结构体长得很像,但两者还是有本质区别的,在使用时请区别对待。