什么是补码加减运算溢出,判别溢出有哪几种方法,说明工作原理是什么?
溢出,是通用的词汇,其含义,还是很容易理解的。
在计算机中,溢出,专用来说明:“补码运算结果超出预定的范围”。
八位的二进制,作为补码,其表示范围是:-128 ~ +127。
示意图如下:
当你的运算结果,小于-128 时,就会落入 +127 下面的范围,成为正数。
当你的运算结果,大于+127 时,就会落入-128 上面的范围,成为负数。
根据这些表现,你就可以找到两种“判别溢出”的方法。
一、两个数值相加,和,超出了-128~+127,必然就溢出。
二、两个补码相加,和的符号,不符合正常的逻辑关系,这就是溢出了。
另外,你还可以根据 OF 的值,来判断是否溢出。
此外,还有“在二进制运算过程中“进行判断的方法。
这方法,是由 CPU 在内部实现的,判断的结果,即在 OF 上给出。
作为人类,掌握这种方法,并无实际意义。
对于加法来说,操作数符号可能是:
1. [正] + [正]
2. [正] + [负]
3. [负] + [正]
4. [负] + [负]
对于减法来说:
1. [正] - [正] 相当于加法里的(2)
2. [正] - [负] 相当于加法里的(1)
3. [负] - [正] 相当于加法里的(4)
4. [负] - [负] 相当于加法里的(3)
加法里,哪种操作会溢出?
对于一个有N位二进制的补码,其能表示的数字的范围是:
-2^(N-1) ~ +2^(N-1)-1
负数:-2^(N-1) 到 -1
正数:1 到 2^(N-1)-1
那么对于加法来说:
“[正] + [负]”或者“[负] + [正]”永远都不会溢出,不管两个操作数取值多少,结果都落在有效范围内。
而超出有效范围的两种可能性就是:
1. [正] + [正]
2. [负] + [负]
那么再看这两种操作如果超出范围的话,表现形式是什么样的:
定义两个事件:
1. 最高有效位进位:对于N位补码,第N-1位发生了进位;
2. 符号位进位:对于N位补码,第N位符号位发生;
定义4个边界值,以8位有效字长为例:
最大正127 (0111 1111)
最小正1 (0000 0001)
最小负-128(1000 0000)
最大负-1 (1111 1111)
列出所有组合(消除交换律重复)
符号位 最高有效位 是否发生溢出 最高有效位进位 符号位进位
[最小正] + [最小正] 0 0 0 0 N N N
[最小正] + [最大正] 0 0 0 1 Y Y N
[最小正] + [最小负] 0 1 0 0 N N N
[最小正] + [最大负] 0 1 0 1 N Y Y
[最大正] + [最大正] 0 0 1 1 Y Y N
[最大正] + [最小负] 0 1 1 0 N N N
[最大正] + [最大负] 0 1 1 1 N Y Y
[最小负] + [最小负] 1 1 0 0 Y N Y
[最小负] + [最大负] 1 1 1 0 Y N Y
[最大负] + [最大负] 1 1 1 1 N Y Y
所以,从规律上看凡是发生溢出操作,“最高有效位进位”和“符号位进位”必然是一个是Y另一个是N,凡是不发生溢出的操作,必然是两个全Y或者两个全N
这就是根据最高有效位和符号位发生进位的异或来判断是否溢出规律。
下面是逻辑证明环节,很绕,证明方法很简单,就是把几种条件都列出来判断一下即可:
[正]+[正]溢出,因为两个都是正整数,所以符号位都是0,溢出时最高位发生了进位,但因为符号位是0,0加上进位的1是永远不会发生进位的。所以对于前面定义的两个事件:“最高有效位进位”发生了,“符号位进位”未发生。
[正]+[负]不溢出时,此时设正数范围包括0:
如果结果为负整数,因为第二个操作数的符号位是1,结果的符号位也是1,那么“符号位进位”未发生,同时“最高有效位进位”也未发生。因为如果发生了,会产生一个进位1,与符号位1相加会倒是符号位变化,而此时符号位是不变的,所以可以肯定两个事件都没有发生。
如果结果为正整数,那么必然同时发生了“最高有效位进位”和符号位进位”,因为第二个操作数的符号位是1,但结果符号位是0,所以符号位有进位(1 + 0 + X = 10)。而第一个操作数的符号位又是0,所以必然是最高有效位产生了一位进位才让符号位获得了一个1才能进位,对于符号位的计算来说[正]符号位0 + [负]符号位1 + 最高有效位进位X = 10,X必然是1
[负]+[负]溢出,因为两个都是负整数,所以符号位都是1,溢出时符号位发生了进位(两个都是1)。再看最高有效位是否有进位,对于8位的负数加法来说,产生溢出,意味着结果范围在-129~-256之间,这个范围内的二进制数,以9位补码来看,就是1 0111 1111到1 0000 0000,那么这两个数如果减掉符号位相加的1 0000 0000的话,得到的结果就是 0111 1111到0000 0000,这个范围内的数字第8位都是0,所以从符号位的计算可以得知[负]符号位1 + [负]符号位1 + 最高有效位进位X = 10,最高有效位的进位值必然是0,所以最高有效位必然没有发生进位。
参考资料
不仅是补码运算,任何形式的运算,都可能产生溢出。
比如:1999 年、2000 年 ...,这是用四位十进制表示。
当到了 9999 年,再过一年,你如果还用四位表示,这就溢出了。
字长为 8 位二进制,补码的范围是-128 至+127。
那么,+125 + 4 = +129,这就溢出了。