这两天整结构体指针强制类型转换的问题,真好碰到一个网友的一个问题,我把这个问题实例化实现, 发现这个实例对弄清楚下面三个问题有很大帮助:
1,结构体指针强制类型转换的问题
2,char字符cout输出的问题
3,结构体数据对齐的问题 具体下面的实例有分析
#include <iostream>
using namespace std;
#pragma pack(1)
typedef struct msg{
unsigned char id;
unsigned char cmd;
unsigned char byte[4];
} MSG;
typedef struct cemsg{
unsigned char id;
unsigned char cmd;
unsigned char data[10];
} CEMSG;
typedef struct msg_p{
unsigned char id;
unsigned char cmd;
unsigned char* ptr;
} MSG_P;
int main(){
/****** 注意:
1. 用void*输出char变量的地址,对于char变量的应用的cout输出,
认为是char*,输出它的 内容
2. 从本次测试来看,指针的类型转换,其实是一种虚拟转换,只是用新结构来解析老结构
在本例中可以明显体会。
*****/
void* addr;
MSG msgTemp = { 0x0c, 0x11, {0xD4,0x00,0xFF,0x00} };
MSG* d_OsMsgPtr = &msgTemp; MSG_P* d_p;
/*CEMSG和MSG都是数组,数据对齐没有空缺,转换没有问题
CEMSG* d_p;
d_p = (CEMSG*) d_OsMsgPtr; */
cout << "size of MSG : " << sizeof(MSG)
<< "size of CEMSG : " << sizeof(CEMSG) << "--" << sizeof(unsigned char*)
<< "size of MSG_P : " << sizeof(MSG_P) << endl;
cout << "msgTemp : " << " id = " << msgTemp.id
<< " cmd = " << msgTemp.cmd
<< " byte = " << msgTemp.byte << endl;
cout << "d_OsMsgPtr : " << d_OsMsgPtr->id
<< d_OsMsgPtr->cmd << d_OsMsgPtr->byte << endl;
cout << "d_OsMsgPtr's addr of byte" ;
addr = &d_OsMsgPtr->byte[0];
cout << " byte[0] : " << addr ;
addr = &d_OsMsgPtr->byte[1];
cout << " byte[1] : " << addr ;
addr = &d_OsMsgPtr->byte[2];
cout << " byte[2] : " << addr ;
addr = &d_OsMsgPtr->byte[3];
cout << " byte[3] : " << addr << endl;
d_p = (MSG_P*) d_OsMsgPtr;
cout << "d_p : " << d_p->id << d_p->cmd << endl;
addr = &(d_p->ptr);//指向byte[3]的地址
/*****************************************************************
1,d_p->ptr本身的地址对应byte的地址,而byte中的值才是ptr指向的地址
2,而此处并不是&(d_P->ptr)并不是完全对应byte的地址,而是有两个偏移
这是由于结构体的数据对齐的原因,sizeof(MSG)是6,sizeof(MSG_P)则
是8,MSG_P的结构体里有2个byte的空缺
************ ******************* *****
*d_OsMsgPtr* ----> *id ** id ** <----*d_p*
************ *cmd ** cmd ** *****
*byte[0] ** **
*byte[1] ** **
*byte[2] **ptr **
MSG *byte[3] ** **
*********** **
** ** MSG_P
**********
指针类型转换图
**********************************************************************/
cout << "d_p addr of byte" << "the addr of &(d_p->ptr) : " << addr << endl;
//ptr的内容,即ptr指向目标地址值,此处为byte[2]和byte[3]和两个未知byte的值,
//按地址格式输出(4321)
addr = d_p->ptr; cout << "the addr of d_p->ptr : " << addr << endl;
addr = &d_p->ptr[0];//为上述ptr指向目标值,也是未知值
cout << "the addr of d_p->ptr[0] :" << addr << endl;
return 0;
//证明是由于数据对齐引起转换偏移的,可以加#pragma pack(1),使之不偏移
}