char类型转换为uint32_t/uint16_t/uint8_t/void*类型再分析

更正:参考该文,对于一个字节的十六进制数来数,并没有越限。以下内容仅作为参考,思考过程较为复杂。

我们在该文中已经分析了关于字符指针强制类型转换为uint16_t/uint8_t型指针,不会越限的现象,这里我们做一下具体分析:

    char ch[4]={0xAA,0x11};
    printf("%d  %d\n",*ch,*(ch+1));
    printf("%x  %x\n",*ch,*(ch+1));
    printf("uint8_t:%d  %d\n",*(uint8_t*)ch,*(uint8_t*)(ch+1));//十进制
    printf("uint8_t:%x  %x\n",*(uint8_t*)ch,*(uint8_t*)(ch+1));//十六进制
    printf("uint16_t:%d  %d\n",*(uint16_t*)ch,*(uint16_t*)(ch+1));//十进制
    printf("uint16_t:%x  %x\n",*(uint16_t*)ch,*(uint16_t*)(ch+1));//十六进制


    qDebug()<<"****************************************";
    char j =0xAA;                //170 = 128+42 //-(128-42) = -86;
    void *q  = (void*)j;         //1111 1111 1111 1111 1111 1111 1010 1010
    uint8_t m = (uint8_t)j;        //1010 1010
    uint16_t n = (uint16_t)j;        //1111 1111 1010 1010
    uint32_t l = (uint32_t)j;        //1111 1111 1111 1111 1111 1111 1010 1010
    qDebug()<<"char:"<<j;
    qDebug()<<"void*:"<<q;
    qDebug()<<"uint8_t:"<<m;
    qDebug()<<"uint16_t:"<<n;
    qDebug()<<"uint32_t:"<<l;

运行结果: 

 分析

1)0xAA 存储在char 类型中,发生越限,实际上是存储的是 -86。

printf("%x  %x\n",*ch,*(ch+1)); //相当于把一个字节的数据转换成4个字节的数据:即是-86的四字节十六进制数。

2)86的二进制: 0101 0110------->转换成4字节:0000 0000  0000 0000 0000 0000 0101 0110 

3)-86的二进制:1010 1010   ---->转换成4字节:1111  1111   1111  1111  1111 1111  1010 1010(符号位不变,取反+1)

4)转换成uint8_t  :相当于把  1010 1010  当成一个uint8_t ,即是AA/170;

5)printf("uint16_t:%d ",*(uint16_t*)ch);

主机字节序:低地址存低位,高地址存高位----> 11aa (aa转-86 )--->二进制: 0001 0001 1010 1010--->转uint16_t(11aa,仍然是11aa).

6)转换成(void*)实际上是转换成 uint32_t

7)  char j =0xAA;                //170 = 128+42 //-(128-42) = -86;//1010 1010
    void *q  = (void*)j;         //1111 1111 1111 1111 1111 1111 1010 1010
    uint8_t m = (uint8_t)j;        //1010 1010
    uint16_t n = (uint16_t)j;        //1111 1111 1010 1010
    uint32_t l = (uint32_t)j;        //1111 1111 1111 1111 1111 1111 1010 1010

可以看到,实际上是将-86扩展为int8_t,int16_t ,int32_t之后,然后 再直接取成uint8_t,uint16_t ,uint32_t的。

最正确的写法是,使用uchar ,而不是char,如下程序所示:

     uchar ch[4]={0xAA,0x11};    //-----》使用uchar
    printf("%d  %d\n",*ch,*(ch+1));
    printf("%x  %x\n",*ch,*(ch+1));
    printf("uint8_t:%d  %d\n",*(uint8_t*)ch,*(uint8_t*)(ch+1));//十进制
    printf("uint8_t:%x  %x\n",*(uint8_t*)ch,*(uint8_t*)(ch+1));//十六进制
    printf("uint16_t:%d  %d\n",*(uint16_t*)ch,*(uint16_t*)(ch+1));//十进制
    printf("uint16_t:%x  %x\n",*(uint16_t*)ch,*(uint16_t*)(ch+1));//十六进制


    qDebug()<<"****************************************";

    uchar j =0xAA;            //-----》使用uchar
    void *q  = (void*)j;     //0000 0000 0000 0000 0000 0000 1010 1010
    uint8_t m = (uint8_t)j;        //1010 1010
    uint16_t n =(uint16_t)j;        //0000 0000 1010 1010
    uint32_t l = (uint32_t)j;    //0000 0000 0000 0000 0000 0000 1010 1010
    qDebug()<<"char:"<<j;
    qDebug()<<"void*:"<<q;
    qDebug()<<"uint8_t:"<<m;
    qDebug()<<"uint16_t:"<<n;
    qDebug()<<"uint32_t:"<<l;

运行结果:  

 

  • 1
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值