C语言中不同的结构体类型的指针间的强制转换详解
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
C语言中不同类型的结构体的指针间可以强制转换,很自由,也很危险。只要理解了其内部机制,你会发现C是非常灵活的。
一.
结构体声明如何内存的分布,
结构体指针声明结构体的首地址,
结构体成员声明该成员在结构体中的偏移地址。
变量的值是以二进制形式存储在内存中的,每个内存字节对应一个内存地址,而内存存储的值本身是没有整型,指针,字符等的区别的,区别的存在是因为我们对它们有不同的解读,param的值就是一个32位值,并且存储在某个内存单元中,通过这个32位值就能找到param所指向的结构的起始地址,通过这个起始地址和各个结构所包含变量离起始地址的偏移对这些变量进行引用,
param->bIsDisable只是这种引用更易读的写法,只要param是指向
PAINT_PARAM的指针,那么param的值就肯定存在,param存在,偏移量已知,那么param->bIsDisable就肯定存在,只是要记住,param->bIsDisable只是代表了对param一定偏移地址的值。
不是说某个地址有那个结构体你才能引用,即使没有,你也能引用,因为你已经告诉了编译器param变量就是指向一个PAINT_PARAM结构体的变量并且指明了param的值,机器码的眼中是没有数据结构一说的,它只是机械的按照
指令的要求从内存地址取值,那刚才的例子来说,peg->x,peg->y的引用无论
0x30000000是否存在一个eg结构体都是合法的,如果0x30000000开始的8
个字节存在eg结构体,那么引用的就是这个结构体的值,如果这个位置是未定义的值,那么引用的结果就是这8个字节中的未定义值,内存位置总是存在的,而对内存中值的引用就是从这些内存位置对应的内存单元取值。
举个例子:
typedefstruct_eg
{
int x;
int y;
}eg;
int point = 0x30000000;
eg *peg = (eg*)point;
可以看到point本身只是个整型变量,但是这样的赋值是合法的,peg->x的值是0x30000000开始的四字节,peg->y是0x30000004开始的四字节
pMsg->wParam的值是0x30000000也就是param指向了以0x30000000为首地址的一片内存单元,这片内存单元以PAINT_PARAM 的方式分布
举个例子:
typedefstructQueueNode
{
structQueueNode * pNext;
}tQueueNode;
typedefstruct QMSG
{
tQueueNode Node;
tChatSysMsg data;
}tQMSG;
typedefstructChatSysMsg
{
int Connfd;
char Version;
char MsgType;
char SerialNumber;
int MsgLen;
char Msg[MAX_NUM_STR];
}tChatSysMsg;
它们间的强制转换:
tQMSG * pTempMsg;
(tQueueNode **)&pTempMsg;
&pTempMsg->data 也就是(tChatSysMsg *)&pTempMsg->data;