zoukankan      html  css  js  c++  java
  • 原码,补码表示

    在计算机内,定点数有3种表示法:原码、反码和补码

    所谓原码就是前面所介绍的二进制定点表示法,即最高位为符号位,“0”表示正,“1”表示负,其余位表示数值的大小。

    反码表示法规定:正数的反码与其原码相同;负数的反码是对其原码逐位取反,但符号位除外

    补码表示法规定:正数的补码与其原码相同;负数的补码是在其反码的末位加1。

     

    正数的反码和补码都是和原码相同。

    负数的反码是将其原码除符号位之外的各位求反
    [-3]反=[10000011]反=11111100
    负数的补码是将其原码除符号位之外的各位求反之后在末位再加1。
    [-3]补=[10000011]补=11111101
    一个数和它的补码是可逆的。

     

    有原码就可以了,为什么还需要反码和补码?

     

    反码是用来算补码的,原码和补码都是用在CPU的基本运算里的,比如数据类型是short:  
      计算5   -   2,并由于实际上CPU没有实现减法电路(注:计算机的硬件结构中只有加法器,所以大部分的运算都必须最终转换为加法,原码没有办法做减法,而在我们使用 的汇编、C等其他高级语言中使用的都是原码,原码转换成补码都是在计算机的最底层进行的)。原码计算是   5+(-2)

         0101

      +1010

      -------  

          1111 


      =-7?显然出错
     所以不管正数还是负数,都使用补码来表示(正数原码和补码是一样的),  2的补码是1110,然后用5补   +   2补  
          0101  
      +  1110  
      ------  
          0011   

      =3,正确
    所以理论上(也仅仅是理论上)我们只要让减数通过一个求反电路,再通过一个+1电路,然后通过加法电路就可以实现减法了。  

    所以补码的设计目的是:
    ⑴使符号位能与有效值部分一起参加运算,从而简化运算规则.
    ⑵使减法运算转换为加法运算,进一步简化计算机中运算器的线路设计

     

     

    原码和反码在表示数的时候的不唯一性,比如表示零的时候,原码就有两种表示法:

         [-0]原=10000000

         [+0]原=00000000

    反码也有两种表示法:

         [+0]反=00000000

         [- 0]反=11111111

    而补码则只有一种[-0]补=00000000。

     

    网上拷贝一些深入介绍其概念资料

    1、原码、反码和补码的表示方法

    (1)原码:在数值前直接加一符号位的表示法。

         例如:    符号位   数值位

          [+7]原=    0     0000111   B

          [-7]原=    1     0000111   B

        注意:
          a. 数0的原码有两种形式:

             [+0]原=00000000B     [-0]原=10000000B

          b. 8位二进制原码的表示范围 :(-127~-0 +0~127)共256个.

    (2)反码:

          正数:正数的反码与原码相同。

          负数:负数的反码,符号位为“1”,数值部分按位取反。

    例如: 符号位    数值位

          [+7]反=   0    0000111   B

          [-7]反=   1    1111000   B

    注意:a. 数0的反码也有两种形式,即

         [+0]反=00000000B

         [- 0]反=11111111B

          b. 8位二进制反码的表示范围:(-127~-0 +0~127)共256个. 

    3)补码的表示方法

    1)模的概念:把一个计量单位称之为模或模数。例如,时钟是以12进制进行计数循环的,即以12为模。在时钟上,时针加上(正拨)12的整数位或减 去(反拨)12的整数位,时针的位置不变。14点钟在舍去模12后,成为(下午)2点钟(14=14-12=2)。从0点出发逆时针拨10格即减去10小 时,也可看成从0点出发顺时针拨2格(加上2小时),即2点(0-10=-10=-10+12=2)。因此,在模12的前提下,-10可映射为+2。由此 可见,对于一个模数为12的循环系统来说,加2和减10的效果是一样的;因此,在以12为模的系统中,凡是减10的运算都可以用加2来代替,这就把减法问 题转化成加法问题了(注:计算机的硬件结构中只有加法器,所以大部分的运算都必须最终转换为加法)。10和2对模12而言互为补数。

    同理,计算机的运算部件与寄存器都有一定字长的限制(假设字长为8),因此它的运算也是一种模运算。当计数器计满8位也就是256个数后会产生溢 出,又从头开始计数。产生溢出的量就是计数器的模,显然,8位二进制数,它的模数为28=256。在计算中,两个互补的数称为“补码”。

    2)补码的表示: 正数:正数的补码和原码相同。

        对于机器数为正数,则[X=[X

         对于机器数为负数,则有[X=[[X

         负数:负数的补码则是符号位为“1”,数值部分按位取反后再在末位(最低位)加1。也就是“反码+1”。

    例如:   符号位 数值位

    [+7]补=    0    0000111   B

    [-7]补=    1    1111001   B

    补码在微型机中是一种重要的编码形式,请注意:

    a.采用补码后,可以方便地将减法运算转化成加法运算,运算过程得到简化。正数的补码即是它所表示的数的真值,而负数的补码的数值部份却不是它所表示的数的真值。采用补码进行运算,所得结果仍为补码。

    b.与原码、反码不同,数值0的补码只有一个,即[0]补=00000000B。

    c.若字长为8位,补码中用(-128)代替了(-0),所以补码的表示范围为: (-128~0~127)共256个;进行补码运算时,应注意所得结果不应超过补码所能表示数的范围。

    注意:(-128)没有相对应的原码和反码, (-128) = (10000000)

     

    使用补码, 不仅仅修复了0的符号以及存在两个编码的问题, 而且还能够多表示一个最低数. 这就是为什么8位二进制, 使用原码或反码表示的范围为[-127, +127], 而使用补码表示的范围为[-128, 127].

    因为机器使用补码, 所以对于编程中常用到的32位int类型, 可以表示范围是: [-231, 231-1] 因为第一位表示的是符号位.而使用补码表示时又可以多保存一个最小值.

    四 原码, 反码, 补码 再深入

    计算机巧妙地把符号位参与运算, 并且将减法变成了加法, 背后蕴含了怎样的数学原理呢?

    将钟表想象成是一个1位的12进制数. 如果当前时间是6点, 我希望将时间设置成4点, 需要怎么做呢?我们可以:

    1. 往回拨2个小时: 6 - 2 = 4

    2. 往前拨10个小时: (6 + 10) mod 12 = 4

    3. 往前拨10+12=22个小时: (6+22) mod 12 =4

     

    2,3方法中的mod是指取模操作, 16 mod 12 =4 即用16除以12后的余数是4.

    所以钟表往回拨(减法)的结果可以用往前拨(加法)替代!

    现在的焦点就落在了如何用一个正数, 来替代一个负数. 上面的例子我们能感觉出来一些端倪, 发现一些规律. 但是数学是严谨的. 不能靠感觉.

    首先介绍一个数学中相关的概念: 同余

    同余的概念

    两个整数a,b,若它们除以整数m所得的余数相等,则称a,b对于模m同余

    记作 a ≡ b (mod m)

    读作 a 与 b 关于模 m 同余。

     

    举例说明:

    4 mod 12 = 4

    16 mod 12 = 4

    28 mod 12 = 4

    所以4, 16, 28关于模 12 同余.

    负数取模

    正数进行mod运算是很简单的. 但是负数呢? 

    下面是关于mod运算的数学定义:

     

    clip_image001

    上面是截图, "取下界"符号找不到如何输入(word中粘贴过来后乱码). 下面是使用"L"和"J"替换上图的"取下界"符号:

    x mod y = x - y L x / y J

    上面公式的意思是:

    x mod y等于 x 减去 y 乘上 x与y的商的下界.

    以 -3 mod 2 举例:

    -3 mod 2

    = -3 - 2xL -3/2 J

    = -3 - 2xL-1.5J

    = -3 - 2x(-2)

    = -3 + 4 = 1

    所以:

    (-2) mod 12 = 12-2=10

    (-4) mod 12 = 12-4 = 8

    (-5) mod 12 = 12 - 5 = 7

    开始证明

    再回到时钟的问题上:

    回拨2小时 = 前拨10小时

    回拨4小时 = 前拨8小时

    回拨5小时= 前拨7小时

    注意, 这里发现的规律!

    结合上面学到的同余的概念.实际上:

    (-2) mod 12 = 10

    10 mod 12 = 10

    -2与10是同余的.

    (-4) mod 12 = 8

    8 mod 12 = 8

    -4与8是同余的.

    距离成功越来越近了. 要实现用正数替代负数, 只需要运用同余数的两个定理:

    反身性:

    a ≡ a (mod m)

    这个定理是很显而易见的.

    线性运算定理:

    如果a ≡ b (mod m),c ≡ d (mod m) 那么:

    (1)a ± c ≡ b ± d (mod m)

    (2)a * c ≡ b * d (mod m)

    如果想看这个定理的证明, 请看:http://baike.baidu.com/view/79282.htm

    所以:

    7 ≡ 7 (mod 12)

    (-2) ≡ 10 (mod 12)

    7 -2 ≡ 7 + 10 (mod 12)

    现在我们为一个负数, 找到了它的正数同余数. 但是并不是7-2 = 7+10, 而是 7 -2 ≡ 7 + 10 (mod 12) , 即计算结果的余数相等.

    接下来回到二进制的问题上, 看一下: 2-1=1的问题.

    2-1=2+(-1) = [0000 0010] + [1000 0001]= [0000 0010] + [1111 1110]

    先到这一步, -1的反码表示是1111 1110. 如果这里将[1111 1110]认为是原码, 则[1111 1110]原 = -126, 这里将符号位除去, 即认为是126.

    发现有如下规律:

    (-1) mod 127 = 126

    126 mod 127 = 126

    即:

    (-1) ≡ 126 (mod 127)

    2-1 ≡ 2+126 (mod 127)

    2-1 与 2+126的余数结果是相同的! 而这个余数, 正式我们的期望的计算结果: 2-1=1

    所以说一个数的反码, 实际上是这个数对于一个膜的同余数. 而这个膜并不是我们的二进制, 而是所能表示的最大值! 这就和钟表一样, 转了一圈后总能找到在可表示范围内的一个正确的数值!

    而2+126很显然相当于钟表转过了一轮, 而因为符号位是参与计算的, 正好和溢出的最高位形成正确的运算结果.

    既然反码可以将减法变成加法, 那么现在计算机使用的补码呢? 为什么在反码的基础上加1, 还能得到正确的结果?

    2-1=2+(-1) = [0000 0010] + [1000 0001] = [0000 0010] + [1111 1111]

    如果把[1111 1111]当成原码, 去除符号位, 则:

    [0111 1111] = 127

    其实, 在反码的基础上+1, 只是相当于增加了膜的值:

    (-1) mod 128 = 127

    127 mod 128 = 127

    2-1 ≡ 2+127 (mod 128)

    此时, 表盘相当于每128个刻度转一轮. 所以用补码表示的运算结果最小值和最大值应该是[-128, 128].

    但是由于0的特殊情况, 没有办法表示128, 所以补码的取值范围是[-128, 127]

     

    参考:http://www.cnblogs.com/zhangziqiu/archive/2011/03/30/ComputerCode.html

    http://blog.csdn.net/macky0668/article/details/3917878

  • 相关阅读:
    PHP、JAVA、C#、Object-C 通用的DES加密
    xtraScrollableControl 滚动条随鼠标滚动
    C#让TopMost窗体弹出并置顶层但不获取当前输入焦点的终极办法
    C#获取“所有用户桌面”的路径
    C#如何获取快捷方式指向的目标文件
    10
    09
    新浪微博中tableview中头部信息
    ASIHTTPRequest类库简介和使用说明
    IOS常用设计模式之委托模式
  • 原文地址:https://www.cnblogs.com/balingybj/p/4685480.html
Copyright © 2011-2022 走看看