1
下面代码的运行结果为?
int main()
{
unsigned char puc[4];
struct tagPIM
{
unsigned char ucPim1;
unsigned char ucData0 : 1;
unsigned char ucData1 : 2;
unsigned char ucData2 : 3;
}*pstPimData;
pstPimData = (struct tagPIM*)puc;
memset(puc,0,4);
pstPimData->ucPim1 = 2;
pstPimData->ucData0 = 3;
pstPimData->ucData1 = 4;
pstPimData->ucData2 = 5;
printf("%02x %02x %02x %02x\n",puc[0], puc[1], puc[2], puc[3]);
return 0;
}
正确答案:
02 29 00 00
解析:
想要回答正确这道题,你必须要对位段有一定的了解(如果忘记了,快去复习吧~),接下来我们一步步分析
我们先看结构体:
struct tagPIM
{
unsigned char ucPim1;
unsigned char ucData0 : 1;
unsigned char ucData1 : 2;
unsigned char ucData2 : 3;
}*pstPimData;
首先思考一下这个问题:这个结构体占用了几个字节?
让我们把图画出来:
答案是2个字节
unsigned char puc[4];
给了我们4个字节
pstPimData = (struct tagPIM*)puc;
结构体只占用2个字节
知道了这些,我们再看赋值
pstPimData->ucPim1 = 2;
pstPimData->ucData0 = 3;
pstPimData->ucData1 = 4;
pstPimData->ucData2 = 5;
因为 ucPim1 把一个字节占据了,所以
因为 ucData0 = 3 ,而 3 转换成二进制为 11 ,而又因为 ucData0 只分配到了一个比特位,于是越界了,发生截断,只保留了后一位 1
因为 ucData1 = 4 ,而 4 转换成二进制为 100 ,而又因为 ucData1 只分配到了两个比特位,于是越界了,发生截断,只保留了后两位 00
大家根据以上过程来想一想 ucData2 = 5 的结果吧。
最终 puc[1] 如下:
最后用16进制打印出 puc 的每个元素
其中 puc[0] 不用多说,就是 2
puc[1] 为 29
计算过程如下:
如果看不懂的话,就去复习进制转换吧~
puc[2] 和 puc[3] memset 后都动过,不必多说,肯定是00
嘿嘿,这道题就做出来了 ^_^
2
有以下宏定义和结构定义
#define MAX_SIZE A+B
struct _Record_Struct
{
unsigned char Env_Alarm_ID : 4;
unsigned char Para1 : 2;
unsigned char state;
unsigned char avail : 1;
}*Env_Alarm_Record;
struct _Record_Struct *pointer = (struct _Record_Struct*)malloc(sizeof(struct _Record_Struct) * MAX_SIZE);
当 A=2,B=3时,pointer 分配几个字节的空间?
正确答案:
9
在做这道题时要注意,#define 执行的是查找替换
让我们一步一步来分析
首先看结构体
struct _Record_Struct
{
unsigned char Env_Alarm_ID : 4;
unsigned char Para1 : 2;
unsigned char state;
unsigned char avail : 1;
}*Env_Alarm_Record;
它占几个字节呢?
答案是 3
原因:
结构体向最长的 char 对齐,前两个位段元素一共 4+2 位,不足8位,合起来占 1 字节,最后一个单独 1 字节,一共 3 字节。
由于 #define 执行的是查找替换, sizeof(struct _Record_Struct) * MAX_SIZE这个语句其实是 3*2+3 ,结果为9