原码、反码、补码(计算机底层)

原码:十进制数据的二进制表型形式,最左边是符号位,0为正,1为负。

反码:正数的补码、反码是其本身,负数的反码是符号位保持不变,其余位取反。

补码:正数的补码是其本身,负数的补码是在其反码的基础上+1。

原码

十进制数据的二进制表现形式,最左边为符号位,0为正,1为负。

利用原码对正数进行计算是不会有问题的。

但是如果是负数计算,结果就会出错,实际运算的结果,跟我们预期的结果是相反的。

比如56的原码为:

一个0或者一个1叫bit(比特位)。

八个bit为一组叫一个字节(是计算机最小的存储单位)。

一个字节最大值为:127

一个字节最小值为:-127 

原码的数字计算

正数进行计算就是二进制的加一。

 

 原码的弊端

利用原码进行计算的时候,如果是正数完全没有问题。

但是如果是负数计算,结果就会出错,实际运算的方向,跟正确的运算方向是相反的。

例如,正常0 + 1要等于1,但是实际上10000001的值是-1;-1 + 1正常为0,但是实际上是-2;在-2的基础上+1正常为-1,但是实际上这个值为-3。

 

为什么?大家看一下数轴,黄色小狗使我们预期的效果,黑色的小狗是实际的效果。我们正常都是认为从1000000变成10000001这个过程是加一,但是因为是负数的原因所以是减一,所以结果与我们的预期相反。

 为了解决这个问题,有个大聪明提出了,我们把这个数轴反过来不就好了。

反码

反码出现的目的

为了解决原码不能计算负数的问题而出现的。

计算规则

正数的反码不变,负数的反码在原码的基础上,符号位不变。数值取反,0变1,1变0。

我们看一下-56:

-56 + 1 = -55

反码的弊端

负数在运算的时候,如果结果不跨0,是没有任何问题的,但是如果结果跨0,跟实际结果会有1的偏差。

反码有一个问题如果跨0则计算会出现问题。

例如:-5 + 6 = 00000000,结果还是0。

再看一个-4 + 7,这个结果为2。

为什么呢?因为反码中的0是有两种表现形式11111111(00000000),这导致我们计算过程中,如果没有跨0计算是没有任何问题的,但是如果跨0了,会有一个误差。

补码

补码出现的目的

为了解决负数计算时跨0的问题而出现的。

计算规则

正数的补码不变,负数的补码在反码的基础上+1。(加一的目的就是为了屏蔽正负零的存在。)

另外补码还可能多记录一个特殊的值-128,该数据在1个字节下,没有原码和补码。

我们来看看补码能否真正帮我们完美解决负数计算问题。

不跨零。

跨零(我们以-4 + 5为例)。

由此可见,补码完美解决了正数和负数的计算问题,因此,在计算机中数字的存储和计算都是以补码的形式进行的。

因此一个字节的存储是从-128到127。

补码的注意点

计算机中的存储和计算都是以补码的形式进行的。

扩展

通过上述内容,大家都明白了,原码、反码、补码的内容了。那么在这个基础上我们就可以搞一些有趣的东西了。

基本数据类型

byte类型

1个字节。

所以byte类型的10为:

0000 1010

short类型

2个字节。

所以short类型的10为:

0000 0000 0000 1010

int类型

4个字节。

所以int类型的10为:

0000 0000 0000 0000 0000 0000 0000 1010

long类型

8个字节。

所以long类型的10为:

0000 0000 0000 0000 0000 0000 0000 0000 0000 0000 0000 0000 0000 0000 0000 1010

类型转换

隐式转换

就四个字:前面加零。

public class Main {
    public static void main(String[] args) {
        byte a = 10;//0000 1010
        int b = a;//0000 0000 0000 0000 0000 0000 0000 1010
        System.out.println(b);
    }
}

强制转换

就是去掉多余的比特位。

public class Main {
    public static void main(String[] args) {
        int a = 300;//0000 0000 0000 0000 0000 0001 0010 1100
        byte b = (byte)a;//0010 1010
        int c = 200;//0000 0000 0000 0000 0000 0000 1100 1000
        byte d = (byte)c;//1100 1000
        System.out.println(b);//44
        System.out.println(d);//-56
    }
}

其他运算符

public class Main {
    public static void main(String[] args) {
        int a = 200;//0000 0000 0000 0000 0000 0000 1100 1000
        int b  = 10;//0000 0000 0000 0000 0000 0000 0000 1010
        //最后结果为8
        System.out.println(a & b);//0000 0000 0000 0000 0000 0000 0000 1000
        //最后结果为202
        System.out.println(a | b);//0000 0000 0000 0000 0000 0000 1100 1010
        //左移2次 最后结果为800(左移一次就是乘2)
        System.out.println(a << 2);//00 0000 0000 0000 0000 0000 0000 100000
        //右移2次 最后结果为2(右移一次就是除2)
        System.out.println(b >> 2);//00 0000 0000 0000 0000 0000 0000 0000 10
        //无符号右移2次 最后结果为2
        System.out.println(b >>> 2);//00 0000 0000 0000 0000 0000 0000 0000 10
    }
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值