结构体指针强制类型转换

本文通过实例探讨了结构体指针强制类型转换、字符输出及数据对齐的问题。具体分析了一个包括结构体定义、类型转换、内存布局及数据对齐影响的C++代码段。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

这两天整结构体指针强制类型转换的问题,真好碰到一个网友的一个问题,我把这个问题实例化实现, 发现这个实例对弄清楚下面三个问题有很大帮助:

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),使之不偏移

 }

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值