结构体中的位定义

目录

实验现象

总结


实验现象

以下实验都在x86架构的cpu上完成。在工作中,经常遇到按位(bit)定义结构体 的情况。由于一个字节有8个位,这时,程序员往往对bit的位置产生困惑。现在给出2个例子,来说明位的定义次序。

第一个例子是将unsigned char 分成8个比特。

#pragma pack(push,1)

typedef struct ST_TEST
{
	unsigned char	ucA:1;
	unsigned char	ucB:1;
	unsigned char	ucC:1;
	unsigned char	ucD:1;
	unsigned char	ucE:1;
	unsigned char	ucF:1;
	unsigned char	ucG:1;
	unsigned char	ucH:1;
} ST_TEST;
#pragma pack(pop)

#include <string.h>
#include <stdio.h>
int main(void)
{
	ST_TEST stTest;
	stTest.ucA = 1;
	stTest.ucB = 0;
	stTest.ucC = 0;
	stTest.ucD = 0;
	stTest.ucE = 0;
	stTest.ucF = 0;
	stTest.ucG = 1;
	stTest.ucH = 0;

	unsigned char ucTest;
	memcpy(&ucTest, &stTest, 1);
	//没有现成的打印二进制的方法,所以用16进制打印
	printf("%x", ucTest);

	scanf("%c", &ucTest);
	return 0;
}

结果是0x41,也就是二进制的0b01000001.可见,定义在一开头的ucA反而落到了最后,而倒数第二的ucG起始在左起正数第二。所以,结构体里定义比特,次序起始是反的。


 

第二个例子是将一个int型的变量按比特分成7部分:

#include <stdio.h>
#include <string.h>
#include <stdlib.h>

#pragma pack(push,1)

typedef struct ST_INT
{
	int			m_A : 1;
	int			m_B : 2;
	int			m_C : 3;
	int			m_D : 4;
	int			m_E : 5;
	int			m_F : 6;
	int			m_G : 11;
} ST_INT;

#pragma pack(pop)


int main()
{
	ST_INT stTest;
	stTest.m_A = 1;
	stTest.m_B = 3;
	stTest.m_C = 6;
	stTest.m_D = 0;
	stTest.m_E = 63;
	stTest.m_F = 48;
	stTest.m_G = 1024;

	int iTest;
	memcpy(&iTest, &stTest, sizeof(int));
	char s[1024] = { 0 };
	_itoa(iTest, s, 2);
	printf("%s\n\n\n\n\n", s);

	memcpy(s, &iTest, sizeof(int));
	char arr[1024] = { 0 };
	_itoa((int)s[0], arr, 2);
	printf("%s\n", arr);
	memset(arr, 0, 1024);
	_itoa((int)s[1], arr, 2);
	printf("%s\n", arr);
	memset(arr, 0, 1024);
	_itoa((int)s[2], arr, 2);
	printf("%s\n", arr);
	memset(arr, 0, 1024);
	_itoa((unsigned char)s[3], arr, 2);
	printf("%s\n", arr);
	scanf(arr);

	return 0;
}

结果:

总结

1)定义在结构体后面的数据项,反而出现在高位。

2)虽然数据项m_G跨了11个比特(超过1字节),但是这11个比特仍然按照从高到低的次序排列。

3)如果把ST_INT这个结构体视为一个32比特的二进制数(抛开计算机的变量类型不谈,就是个数学上的数字)的话,写在最后的数据项m_G出现在二进制数权重最高的位置上,接下来是m_F,依次类推,m_E,m_D...直到完成这个二进制数。由于x86cpu采用小端,所以这个32位二进制数被顺序的切分成4段字节,权重最高的字节放到内存的后面,而权重最低的放到内存的前面,但是每个字节内部的比特次序仍保持在32位数中的原始状态。经过这样的变化后的数据才是x86架构计算机的内存里存放的数据。

4) 对于大端的计算机,结构体前面的数据项出现在高位。如果把结构体也视为一个32比特的二进制数,二进制数的四个字节不应像3)那样倒序排列,而是维持已有顺序。

  • 14
    点赞
  • 43
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值