一个小题目,猜猜运行结果:
#include <iostream.h>
int main()
{
char a = 0xd6;
cout << (signed int)0xd6 << endl;
cout << (signed int)a << endl;
cout << (unsigned int)0xd6 << endl;
cout << (unsigned int)a << endl;
a = 0x11;
cout << (signed int)0x11 << endl;
cout << (signed int)a << endl;
cout << (unsigned int)0x11 << endl;
cout << (unsigned int)a << endl;
return 0;
}
正确的运行结果是:
214
-42
214
4294967254
17
17
17
17
不知道你是不是能够很容易的理解,我是费了很大的劲才搞清楚其中是怎么回事的:
在32位的机器上,char占一个字节,int占四个字节
对0xd6本身这个常量来说,它在计算机中的存储二进制串应该是
00000000 00000000 00000000 11010110
所以就不难理解(signed int)0xd6 和 (unsigned int)0xd6的输出结果了。
然后我们把它赋给一个字符变量a,它就变成了一个字节
11010110
当我们用(signed int)a时,它又被强制转换为整形,而整形是有四个字节的,所以需要进行高位补字节,这里我
还有一个较大的疑惑就是在类型转换时进行高位扩展时是按照什么原则来的(不知道这个是由语言本身决定的还是编译器决定的),我通过几个实验在VC6上得出的结论是它是按照高位字节进行扩充。11010110的高位是1,所以(signed int)a的二进制串应该是
11111111 11111111 11111111 11010110
因为它是一个负数,所以这个二进制串是它的补码,不难得出它的原码是
10000000 00000000 00000000 00101010
从而得出它的值是-32
同样(unsigned int)a的二进制串与(signed int)a的二进制串是完全一样的
但因为它是无符号的,不存在原码、反码或补码,所以这个二进制串就可看作它的原码
从而得出它的值是4294967254
同理可分析0x11的结果。
msn:zhanghuanling AT msn.com