关于原码,反码,补码我们必须先明确:
只有有符号数才有原码反码补码,其在内存中是以补码的形式存储, 无符号数在内存中是以二进制数的形式存储的!
所谓有符号数的原码就是将它的最高位当成符号位,0:表示正,1表示负。其余为数值位表示数值的大小。正数的原码反码补码相同。负数的原码除符号位外其余各位等于其绝对值数的原码。负数的反码等于除符号位外对其原码逐位取反。负数的补码在其反码的基础上加1.
下面看两个简单的例子:
(1)
int main(int argc ,char **argv)
{
unsigned char c;
c = 300;
fprintf(stdout, "c = %d/n", c)p;
return 0;
}
结果:c = 44;
这很好理解因为数据溢出300 - 256 刚好为 44;
(2)
int main(int argc, char **argv)
{
signed char c;
c = 200;
fprintf(stdout, "c = %d/n", c);
return 0;
}
结果为:c = -56;
为什么不是:200 - 128 等于72呢?
而是 - 56?
这是因为200无论把它当成无符号数还是有符号数其在内存中存储形式都是11001000.
c = 200,即11001000写入变量c所代表的字节中。因为我们声明了c为有符号数。有符号是在内存中都是以补码的形式存储的。所以编译器把11001000看成补码。减1得到反码11000111.除符号位外各位取反得到原码10111000.转化成十进制恰好位-65。