首页 > 代码库 > 一道int与二进制加减题

一道int与二进制加减题

int dis_data = http://www.mamicode.com/32769;<?xml:namespace prefix = o />

if( dis_data > 0x7fff)  dis_data -= 0xffff;

printf(“%d\n”,dis_data);

 

上面的dis_data 输出值会是多少?  初一看可能还看不出来,那就计算一下:

0x7fff转换为十进制为 32767,显然 dis_data > 0x7fff, 所以要执行 dis_data -= 0xffff;这一句代码。 现在dis_data = http://www.mamicode.com/32769, 那么 0xffff转换为十进制等于多少呢? 如果把0xffff看成是16位,那么 0xffff 转换为十进制显然等于-1,如果把0xffff看出是双字的32位,那么 0xffff转换为十进制为 65535

为什么?

看成16位,0xffff 转换为十进制显然等于-1;

看成32位,那么 0xffff转换为十进制为 65535

 

因为其最高位代表符号位,如果看成是16位,那么就是一个字的大小,即16位。第16位是符号位,后面的15位才是数值位, 所以16位能表示的最大十进制为 0x7fff 即:32767 到最小负数 0xffff -32768,而如果看成是32位,那么0xffff的第16位不是符号位而是数值位,所以其计算结果是 65535.

而在 dis_data  -= 0xffff; 表达式中 dis_dataint型,位双字,4字节,即32位,所以 0xffff也要看成为32位来参与运算。 所以

dis_data - = 0xffff; 就相当于 dis_data = http://www.mamicode.com/dis_data – 0xffff = 32769 – 65535 = -32766

-32766在双字十六进制中的表示为:ffff8002。在单子中直接表示为 8002

所以最终结果是 -32766

 


int  i = 2;

printf(“%d\n”,~++i); //-4

初一看 ~++i 就是先 ++i 那么是3 然后取反就是 -3了, 怎么是-4呢? 这里要注意了, 這里的取反符号 ~ 是指的按位取反,是按二进制位取反,所以++i 等于3之后, 要先把3表示为 二进制数(这里假设是8位)即0000 0011, 然后按位取反就得到 1111 1100。但是这里要注意 这里还不能直接当作最终结果输出,因为这是计算机内存中保存的二进制补码形式, 要正确输出 ~++i 的表示结果,还是要先把补码转换为原码才能输出,当然如果是整数,其正反补都是一样的则可以不用转换直接输出,但是现在这里是负数,所以要先转换为原码, 负数补码转换为原码的规则是: 符号位不变,其余位取反,然后加1 所以先符号位不变其余位取反,则由1111 1100 变为 1000 0011 然后加1 则变为 1000 0100, 这就是原码, 然后转为为十进制表示, 其最高位表示符号位 1 表示负数,然后0100 表示为4 ,所以最终输出结果就是 -4