位域与大小端


大小端字节序介绍见:点击打开链接


大小端字节序指的是多字节类型的字节数据在内存中的存储顺序,字节内的各bit位置并不变化。

例如数据:0x12345678

在大端模式下的存储情况为:

地址0123
十六进制数据0x120x340x560x78
二进制数据0001 00100011 01000101 01100111 1000


在小端模式下的存储情况为:

地址0123
十六进制数据0x780x560x340x12
二进制数据0111 10000101 01100011 01000001 0010

由以上数据可见,大小端情况下只是字节数据的存储位置不同,字节内的各bit位置并不变话。


那什么情况下,大小端模式会影响bit位的存储顺序呢? 答:当结构体内的数据按位域定义时。这种情况下,结构体内按位域定义的数据类型,一定要区分大端字节序和小端字节序定义!


在对Struct(结构体)中成员进行内存分配的时候,“按排列顺序分配,先分配排在前面的”:

1.大端(big endian)模式,从高位向地位分配

对字节,先分配高字节(内存低地址中),再分配低字节(内存高地址中)。

对位域,先分配高bit位,再分配低bit位。

2.小端(little endian)模式,从地位向高位分配

对字节,先分配低字节(内存低地址中),再分配高字节(内存高地址中)。

对位域,先分配低bit位,再分配高bit位。



定义如下结构体:

typedef struct tagBigOrLittleData
{
	UINT32   uiA:5; /*对应数据赋值:00101*/
	UINT32   uiB:9; /*对应数据赋值:000001001*/
	UINT32   uiC:12;/*对应数据赋值:000000001100*/
	UINT32   uiD:6; /*对应数据赋值:000110*/

}BigOrLittleData_S;

定义结构体成员:stData 各个字段赋值如上面注释


大端模式下,stData 数据在内存中的值为:/*00101 000   001001 00  00000011  00 000110*/

地址0123
十六进制数据0x280x240x030x6
二进制数据00101 000
001001 0000000011
00 000110
解释uiA(5)+uiB(H3)uiB(L6)+uiC(H2)uiC(M8) uiC(L2)+uiD(6)


  结构体成员内存分配原则和上面描述一致:“按排列顺序分配,先分配排在前面的成员,在大端模式下,对位域,先对高bit位进行分配,在对低bit位进行分配”。



小端模式下,stData 数据在内存中的值为:/*001 00101   00 000001  00000011  000110 00*/

地址0123
十六进制数据0x250x010x030x18
二进制数据001  00101
00  000001
00000011
000110  00
解释uiB(L3)+uiA(5)uiC(L2)+uiB(H6)uiC(M8)uiD(6) + uiC(H2)

   结构体成员内存分配原则和上面描述一致:“按排列顺序分配,先分配排在前面的成员,在小端模式下,对位域,先对低bit位进行分配,在对高bit位进行分配”。


VS2010 验证代码及结果(小端模式):

typedef struct tagBigOrLittleData
{
	UINT32   uiA:5; /*对应数据赋值:00101*/
	UINT32   uiB:9; /*对应数据赋值:000001001*/
	UINT32   uiC:12;/*对应数据赋值:000000001100*/
	UINT32   uiD:6; /*对应数据赋值:000110*/

}BigOrLittleData_S;

int _tmain(int argc, _TCHAR* argv[])
{


	BigOrLittleData_S  stData = {0};

	stData.uiA = 5;
	stData.uiB = 9;
	stData.uiC = 12;
	stData.uiD = 6;

	/*001 00101   00 000001  00000011  000110 00*/

	getchar();

	return 0;
}





由此可见,同样一个按位域定义的结构体,大小端模式下,数据在内存中的存储顺序是完全不一样的,说白了,就是字节的值和位置都不一样。


所以,如果大端定义了上面的结构体,那么小端对应的结构体应该定义为:

typedef struct tagBigOrLittleData
{
    UINT32   uiD:6; /*对应数据赋值:000110*/
    UINT32   uiC:12;/*对应数据赋值:000000001100*/
    UINT32   uiB:9; /*对应数据赋值:000001001*/
    UINT32   uiA:5; /*对应数据赋值:00101*/

}BigOrLittleData_S;

此时,在小端模式下,stData 数据在内存中的值为:/*00 000110   00000011  001001 00  00101 000*/

地址0123
十六进制数据0x060x030x240x28
二进制数据00 00011000000011001001 0000101 000
解释uiC(L2)+uiD(6)uiC(M8)uiB(L6)+uiC(H2) uiA(5)+uiB(H3)

会发现,结构体各个位域字段“反序”定义以后,小端模式下,字节的值和大端模式的值相同了,但字节序还是反序的(0x06032428 对应 0x28240306),所以在使用时还要进行大小端字节序的转换!


从而,结构体中位域定义的成员,为了兼容大小端,存在以下规则:

1.结构体位域成员的定义,对每个成员的定义要区分大小端,对成员中位域的定义顺序进行翻转。

例如:

typedef struct tagBigOrLittleData
{
#ifdef BIG_ENDIAN
	UINT32   uiA:5; /*对应数据赋值:00101*/
	UINT32   uiB:9; /*对应数据赋值:000001001*/
	UINT32   uiC:12;/*对应数据赋值:000000001100*/
	UINT32   uiD:6; /*对应数据赋值:000110*/

	USHORT   usA:6;
	USHORT   usB:10
#else
	UINT32   uiD:6; /*对应数据赋值:000110*/
	UINT32   uiC:12;/*对应数据赋值:000000001100*/
	UINT32   uiB:9; /*对应数据赋值:000001001*/
	UINT32   uiA:5; /*对应数据赋值:00101*/

	USHORT   usB:10
	USHORT   usA:6;

#endif
}BigOrLittleData_S;


2.数据在不同主机之间传输使用时,要做字节序的转换。(发送时进行主机序到网络序的转换,接收时进行网络序到主机序的转换)

评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值