免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 4199 | 回复: 8
打印 上一主题 下一主题

指针的强制类型转换问题 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2010-09-03 10:25 |只看该作者 |倒序浏览
本帖最后由 dxs187 于 2010-09-03 15:24 编辑
  1. #include<stdio.h>
  2. int main()
  3. {
  4.     unsigned int a=0xFFFFFFF7;
  5.     unsigned char i=(unsigned char)a;
  6.     char *b=(char *)&a;
  7.     printf("%08x,%08x\n",i,*b);
  8.     return 0;
  9. }
复制代码
i的输出结果可以理解,但是*b的结果为fffffff7,不能理解,b不是指向char型的指针么,为什么输出还是int型的,到底*b输出时编译器是怎么工作的?难道不是按所指类型所占字节数来取的?

论坛徽章:
0
2 [报告]
发表于 2010-09-03 11:26 |只看该作者
这个题我的打印不出东西,直接程序崩掉了

printf("%08x,%08x\n",i,*b);不过你觉得你的目的是打印地址还是打印值?

论坛徽章:
0
3 [报告]
发表于 2010-09-03 11:42 |只看该作者
汗ing... 这个程序应该引起segmentation fault错误吧...
     *b 是访问地址为0xffffffff7的内容...  在Linux下内核都无法访问这个地址...
     在windows中... 貌似4GB的线性地址空间的最后2GB留给用户空间... 但恐怕加载程序的时候. 这个地址也是非法的吧... 建议楼主看看编译之后的汇编代码.... 寻找问题....  

     灰常诡异啊...

论坛徽章:
2
CU十二周年纪念徽章
日期:2013-10-24 15:41:34处女座
日期:2013-12-27 22:22:41
4 [报告]
发表于 2010-09-03 14:00 |只看该作者
看你得到结果,运行的程序里面,b的前面少打了个*吧。

论坛徽章:
0
5 [报告]
发表于 2010-09-03 15:24 |只看该作者
本帖最后由 dxs187 于 2010-09-03 15:26 编辑

不好意思,少打了个&,第六行应该是char *b=(char *)&a;请大家继续帮忙,积聚力量啊,非常感谢~

论坛徽章:
0
6 [报告]
发表于 2010-09-03 17:58 |只看该作者
楼主的这个问题起始于指针的强制类型转换并无任何关系...

      printf语句中的*b所表示的值的的确确是0xf7...  可是在执行参数传递的时候... 由于栈的使用, 以x86 32位机器为例...是以4字节整数倍来使用的... 所以一个字节长的0xf7要放到栈里面,就要扩展成四字节...
所以这里发生了符号扩展... 由于char类型是有符号的.. 而0xf7这个值的最高有效位是1.所以就扩展成了
0xfffffff7。所以*b在堆栈中对应的就是这个值了... 而这一切都发生参数传递期间。
而在执行对%08x这个占位符的解析时.这发生在vsprintf(buf, format, args)函数中。
x只是表示输出形式为小写的十六进制而已。在解析对应的args时, *b在栈中对应的内容会以
num = va_arg(args, unsigned int);的形式读出来, 然后再转换为小写十六进制,复制到buf中。
可见...num = 0xfffffff7。转换成字符串形式的十六进制当然也是这个... 所以就得到了楼主的结果...

所以... 这一切都是符号扩展惹的祸...

论坛徽章:
0
7 [报告]
发表于 2010-09-03 18:00 |只看该作者
b指向a ,*b当然是a的内容了!

论坛徽章:
0
8 [报告]
发表于 2010-09-03 18:15 |只看该作者
b指向a ,*b当然是a的内容了!
zd零 发表于 2010-09-03 18:00



    这位兄弟没看俺的解释么... 不要想当然...  你可以自己写点代码测试一下...  

   unsigned int a = 0x80;
  char *b = (char *)&a;  在printf传递*b时经过符号扩展...就变成0xffffff80... 当然输出也是这个。

  如果这样改... unsigned char *b = (char *)&a; 在printf传递*b时由于b是指向无符号字符类型...所以就不会有符号扩展。其内容就是0x00000080,输出也是这个结果...

论坛徽章:
0
9 [报告]
发表于 2010-09-04 12:06 |只看该作者
回复 6# PCliangtao


    谢谢大侠!
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP