正数的原码反码补码是一样的:
[+2] = [00000010]原 =[00000010]反 =[00000010]补
但是对于负数来说,它的原码反码补码就不尽相同:
[-2] =[10000010]原 =[11111101]反 =[11111110]补
对于人来说,可以直接看出原码和十进制数字的关系,而反码和补码还需要其他一些转换,那么为什么设计出反码补码?
因为计算机只有加法,没有减法,因为在做减法的时候,可以认为是加上一个负数,这样可以减少计算机电路复杂度。所以在用原码进行减法的时候会出问题。比如:我要计算1-1,因为计算机没有减法只有加法,所以计算机自动换成1+(-1)
1 - 1 = 1 + (-1) = [00000001]原 + [10000001]原 = [10000010]原 = -2(符号位也参与运算)
而实际上1-1的结果是0
为了解决这个减法的问题,于是想出了反码。
1 - 1 = 1 + (-1) = [0000 0001]原 + [1000 0001]原= [0000 0001]反 + [1111 1110]反 = [1111 1111]反 = [1000 0000]原 = -0
这样用反码计算得出的结果是正确的,但是有个问题:00000000可以代表+0;10000000可以代表-0;但是+0和-0其实是一样的,用2个编码实在是浪费。于是出现了补码解决了0的符号以及两个编码的问题:
1-1 = 1 + (-1) = [0000 0001]原 + [1000 0001]原 = [0000 0001]补 + [1111 1111]补 = [0000 0000]补=[0000 0000]原
这样0用[0000 0000]表示, 而以前出现问题的-0则不存在了.而且可以用[1000 0000]表示-128:但是注意因为实际上是使用以前的-0的补码来表示-128, 所以-128并没有原码和反码表示.(对-128的补码表示[1000 0000]补算出来的原码是[0000 0000]原, 这是不正确的)
使用补码, 不仅仅修复了0的符号以及存在两个编码的问题, 而且还能够多表示一个最低数. 这就是为什么8位二进制, 使用原码或反码表示的范围为[-127, +127], 而使用补码表示的范围为[-128, 127].
总结:反码是为了解决减法运算,补码是为了解决反码产生的±0的问题。