数字的原码,补码,反码,~和‘-’的区别。取反与负号的区别。

一.原码,反码,补码的基础概念和计算方法。

原码,反码,补码只是计算机存储数字的一种方式。
1,原码
原码就是用第一位表示符号位,其余位表示值,比如8位的二进制数

[+1]原 = 0000 0001

[-1]原= 1000 0001

因为第一位为符号位,所以8位二进制数的取值范围为

[1111 1111 , 0111 1111] 即 【-127,+127】

2,反码
反码是在原码的基础上符号位不变,其余的位数取反,比如

[+3]原 = 00000011原 =0111 1100反

[-3]原 = 1000 0011原 =1111 1100反

3,补码
正数的补码为它本身,比如

[+3]原 = 0000 0011原 =0000 0011补

负数的补码为其反码+1

[-3]原 = 1000 0011原 =1111 1100反=1111 1101补

二,为何要用原码,反码,补码

通过上述我们知道计算机有三种编码方式,对于正数三种编码方式的结果都相同

[+3]原 = 0000 0011原 =0000 0011反=0000 0011补

[-3]原 = 1000 0011原 =1111 1100反=1111 1101补

首先, 因为人脑可以知道第一位是符号位, 在计算的时候我们会根据符号位, 选择对真值区域的加减。 但是对于计算机, 辨别"符号位"显然会让计算机的基础电路设计变得十分复杂! 于是人们想出了将符号位也参与运算的方法。
我们知道, 根据运算法则减去一个正数等于加上一个负数, 即: 3-3 = 3 + (-3) = 0 , 所以机器可以只有加法而没有减法, 这样计算机运算的设计就更简单了。
原码将符号位参与运算的计算方法如下

3-3=3+(-3)= 0000 0011原 + 1000 0011原 = 1000 0110原 = -6

如果用原码表示让符号位参与运算,对于减法来说结果显然不正确,这就是计算机内部不适用原码表示一个数的原因。
反码的计算方法如下

3-3=3+(-3)= 0000 0011原 + 1000 0011原 = 0000 0011反 +1111 1100反 = 1111 1111反 = 1000 0000原 = -0

可以看到用反码计算减法结果真值部分是正确的,但数字0却有两种编码方式+0与-0,显然0带符号位没有任何意义。而且会有0000 0000原和1000 0000原两个编码表示0。
补码的出现就解决了0的符号以及两个编码的问题
补码的计算方法如下

3-3=3+(-3)= 0000 0011原 + 1000 0011原 = 0000 0011反 +1111 1100反 = 0000 0011补 + 1111 1101补 =0000 0000补 = 0000 0000原 = 0

这样0用[0000 0000]表示,而以前出现问题的-0则不存在了。而且可以用[1000 0000]表示-128

(-1)+(-127)= 1000 0001原 + 1111 1111原 = 1111 1110反 +1000 0000反 = 1111 1111补 + 1000 0001补 =1000 0000补 = -128

使用补码运算修复了0的编码表示,而且还能表示一个最低数,-128并没有原码和反码表示。8位二进制数表示的范围为[-128,127],也就是[-27,27-1],其他同理。

三.取反~和负数‘-’的区别

首先看代码:

int main(){
int i=-2147483648;
return printf(“%d,%d,%d,%d”,~i,-i,1-i,-1-i);
}
输出结果为:

2147483647,-2147483648,-2147483647,2147483647
这里就涉及到对负数取反,单目运算符“-”的理解,在int占4个字节,大小范围为-2147483648~2147483648,最高位代表符号位,所以第32位为1,这里就需要理解计算机中负数的表示形式:

计算机中的数字都是以补码的形式存放!

正数的原码、反码、补码都相同

5在计算机中原码、反码、补码均为:00000000 00000000 00000000 00000101

负数的反码为对该数的原码除符号位外各位取反,补码为其反码加1:

-5原码为:10000000 00000000 00000000 00000101

反码为:11111111 11111111 11111111 11111010

补码为:11111111 11111111 11111111 11111011

所以-5在计算机中存储形式为11111111 11111111 11111111 11111011转换为十六进制为:0xFFFFFFFB

但是-2147483648比较特殊!

-2147483648原码为:10000000 00000000 00000000 00000000 (表示-0)

反码为:11111111 11111111 11111111 11111111

补码为:00000000 00000000 00000000 00000000

+0的原码、补码、反码均为00000000 00000000 00000000 00000000

如果用-2147483648用补码00000000 00000000 00000000 00000000来表示则和0的补码一样,不能区分开;

所以计算机规定用10000000 00000000 00000000 00000000来表示-2147483648的补码,

32位最小负整数的补码为10000000 00000000 00000000 00000000

计算机中-2147483648取反为:01111111 11111111 11111111 11111111,此为存储正数的补码,换算原码得到数为2147483647

单目运算符“-”为取负,-2147483648取负为2147483648,显然超过int类型范围(-2147483648~2147483647),刚好超过一个就变为-2147483648,所以其对应的补码仍为10000000 00000000 00000000 00000000,

对应补码理解就是-x=~x+1;即补码取反加1,10000000 00000000 00000000 00000000取反加1后任然为10000000 00000000 00000000 00000000,对应数为-2147483648

1-i为(-i)+1,对应补码为:

10000000 00000000 00000000 00000000+00000000 00000000 00000000 00000001=10000000 00000000 00000000 00000001

转换为十进制为:- 2147483647

-1-i为(-i)+(-1),对应补码为:

10000000 00000000 00000000 00000000+11111111 11111111 11111111 11111111=01111111 11111111 11111111 11111111

转换为十进制为: 2147483647

其实也可以从十进制理解:-2147483648-1=-2147483649,超过范围后变为2147483647

转载于:https://www.cnblogs.com/xiaodingmu/p/8215613.html
https://blog.csdn.net/qq_42001403/article/details/100142480

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值