程序员面试宝典中的一个错误 char * b=(char *)&a

本文转自http://blog.csdn.net/race604/article/details/6725475

 

判断下面函数的输出

 

  1. void main()  
  2. 2 {  
  3. 3     unsigned int a = 0xFFFFFFF7;  
  4. 4     unsigned char i = (unsigned char)a;  
  5. 5     char* b = (char*)&a;  
  6. 6   
  7. 7     printf("%08x, %08x\n", i, *b);  
  8. 8 }  
1 void main()
2 {
3     unsigned int a = 0xFFFFFFF7;
4     unsigned char i = (unsigned char)a;
5     char* b = (char*)&a;
6 
7     printf("%08x, %08x\n", i, *b);
8 }

解答: 输出的结果是000000f7, fffffff7。变量i的输出是没有疑问的,unsigned int到unsigned char直接截断,取低字节。对于变量b,在第5行中,书中解释是这一行等价于:

 

  1. unsigned int* p = &a; // p中的内容是的地址,即p指向a  
  2. char* b = (char*)p; // 此处的强制转换只是使b也指向a而已  
  3. // 这里是char类型的指针转换,而不是char类型的转换,影响的只是指针的寻址  
unsigned int* p = &a; // p中的内容是的地址,即p指向a
char* b = (char*)p; // 此处的强制转换只是使b也指向a而已
// 这里是char类型的指针转换,而不是char类型的转换,影响的只是指针的寻址

书中认为,b是指向的内容就是a,所以解释了*b的输出就是a的内容。

 

但是,我做了如下验证:

 

  1. void main()  
  2. 2 {  
  3. 3     unsigned int a = 0xAAAAAAA7;  
  4. 4     unsigned char i = (unsigned char)a;  
  5. 5     char* b = (char*)&a;  
  6. 6     unsigned char* c = (unsigned char*)&a;  
  7. 7   
  8. 8     printf("%08x, %08x, %08x,%08x\n", a, i, *b, *c);  
  9. 9 }  
  1 void main()
  2 {
  3     unsigned int a = 0xAAAAAAA7;
  4     unsigned char i = (unsigned char)a;
  5     char* b = (char*)&a;
  6     unsigned char* c = (unsigned char*)&a;
  7 
  8     printf("%08x, %08x, %08x,%08x\n", a, i, *b, *c);
  9 }

输出的结果是:aaaaaaa7, 000000a7, ffffffa7,000000a7。按照书中的理论,这里*b和*c明显是无法解释的。经过查阅相关的资料,我认为可以这样解释,其实程序的5行和6行也是进行截断。b或者c是指向a的内存,但是只是指向a的低字节内存。也就是0xA7。b是char*类型,所以把0xA7解释为负数,所以*b输出结果为0xffffffa7,等于-89。而c,为unsigned char*类型,*c的输出为167。

 

参考这里:http://www.cppblog.com/aaxron/archive/2011/02/28/140786.aspx,可以知道,在x86机器上,都是小端字节序(litte-endian)。还有这里http://hi.baidu.com/zhangganglei/blog/item/31b2d9d5575b38d151da4bf6.html关于c语言的内存的问题。

 
posted @ 2014-08-27 14:12  琴心剑魄  阅读(3763)  评论(0编辑  收藏  举报