我们先来谈谈溢出的问题,在进行有符号数运算的时候,如结果超过了机器所能表示的范围,则成为溢出。
那么,什么是机器所能表示的范围呢?
比如说:指令运算的结果用8位寄存器存放,那么对于8位的有符号数据,机器所能表示的范围就是-128~127.如果运算结果超出了机器所能表达的范围,将产生溢出。
注意,这里所讲的溢出,只是对有符号数运算而言。下面我们看两个溢出的例子。
mov al,98(62h)
add al,99(63h)
执行后将产生溢出,因为add al,99进行的有符号数运算是:98+99=197,而197超出了机器所能表示的8位有符号数的范围:-128(-80h)~127(7fh).
再来看一个:
mov al,0f0h
add al,088h
执行后,将产生溢出。因为add al,88h进行的有符号数运算是:
al=al+(-120)=(-16)+(-120)=-136
而结果-136超出了机器所能表示的8位有符号数的范围:-128~127
如果在有符号数运算时,发生溢出,那么运算结果将不正确。
补充1:cpu在执行add等指令的时候,就包含了两种含义:无符号数运算和有符号数运算。对于无符号数运算,CPU用CF位来记录是否产生了进位;对于有符号数运算,CPU用OF位来记录是否产生了溢出。
补充2:注意原码和补码的区别,在指令中的数据,都是以补码的形式存在的,而有符号数的范围,如-80h~7fh,这是原码。原码和补码的转化,请记住三句话:
1.补码的最高位为1,表示负数。
2.正数的补码,就是正数本身。
2.正数的补码,取反加1后,为其负数的补码。
3.负数的补码,取反加1后,为其绝对值。
补充3:判断溢出的其他方法
1.两个不同符号数相加,则结果一定不溢出。
2.双符号位判断。所谓双符号位,就是在正数前补0,在负数前补1,即00表示正号,11表示负号。如果进位将会导致符号位不一致,从而检测出溢出。结果的符号位为01时,称为上溢;为10时,称为下溢。
如:62h+63h=00 1010010+00 1010011=01 0100101,符号位为01,上溢。
如:f0h+88h=11 1110000+11 0001000=110 1111000,符号位为10,下溢。