本文转自http://blog.csdn.net/race604/article/details/6725475
判断下面函数的输出
- 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 ", i, *b);
- 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 ", i, *b); 8 }
解答: 输出的结果是000000f7, fffffff7。变量i的输出是没有疑问的,unsigned int到unsigned char直接截断,取低字节。对于变量b,在第5行中,书中解释是这一行等价于:
- unsigned int* p = &a; // p中的内容是的地址,即p指向a
- char* b = (char*)p; // 此处的强制转换只是使b也指向a而已
- // 这里是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 {
- 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 ", a, i, *b, *c);
- 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 ", 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语言的内存的问题。