cmp是比较指令,cmp的功能相当于减法指令,只是不保存结果。cmp指令执行后,将对标志寄存器产生影响。其他相关指令通过识别这些被影响的标志寄存器位来得知比较结果。
cmp指令格式: cmp 操作对象1,操作对象2
功能:计算操作对象1-操作对象2但并不保存结果,仅仅根据计算结果对标志寄存器进行设置。
例如指令cmp ax,ax,做(ax)-(ax)的运算,结果为0,但并不在ax中保存,仅影响flag的相关各位。指令执行后:zf=1.pf=1,sf=0,cf=0,of=0。
下面的指令:
mov ax,
mov bx,
cmp ax,bx
执行后: (ax)=8, zf=0,pf=1,sf=0,cf=0,of=0。
其实,我们通过cmp指令执行后,相关标志位的值就可以看出比较的结果。
cmp ax,bx
如果(ax)=(bx),则(ax)-(bx)=0,所以:zf=1;
如果(ax)≠(bx),则(ax)-(bx)≠0,所以:zf=0;
如果(ax)<(bx),则(ax)-(bx)将产生借位,,所以:cf=1;
如果(ax)≥(bx),则(ax)-(bx)不必借位,所以:cf=0;
如果(ax)>(bx),则(ax)-(bx)既不必借位,结果又不为0,所以:cf=0并且zf=0;
如果(ax)≤(bx),则(ax)-(bx)既可能借位,结果可能为0,所以:cf=1或zf=1;
指令cmp ax,bx的逻辑含义是比较ax和bx中的值,如果执行后:
zf=1,说明(ax)=(bx)
zf=0,说明(ax)≠(bx)
cf=1,说明(ax)<(bx)
cf=0,说明(ax)≥(bx)
cf=0并且zf=0,说明(ax)>(bx)
cf=1或zf=0,说明(ax)≤(bx)
上面所讲的是用cmp进行有符号数比较时,相关标志位对比较结果的记录。如果用cmp来进行有符号数比较时,CPU用哪些标志位对比较结果进行记录。例如
cmp ah,bh
如果(ah)=(bh),则(ah)-(bh)=0, 所以:zf=1
如果(ah)≠(bh),则(ah)-(bh)≠0, 所以:zf=0;
所以,根据cmp指令执行后zf的值,就可以知道两个数据是否相等。
最后以cmp ah,bh为例,总结一下CPU执行cmp指令后,sf和of的值是如何来说明比较的结果的.
1)如果sf=1,而of=0
of=0, 说明没有溢出,逻辑上真正结果的正负=实际结果的正负;
因sf=1, 实际结果为负,所以逻辑上真正的结果为负,所以(ah)<(bh)。
2)如果sf=1,而of=1
of=1, 说明有溢出,逻辑上真正结果的正负≠实际结果的正负;
因sf=1, 实际结果为负。
实际结果为负,而又有溢出,这说明了是由于溢出导致了实际结果为负,简单分析一下,就可以看出,如果因为溢出导致了实际结果为负,那么逻辑上真正的结果必然为正。
这样,sf=1,of=1,说明了(ah)>(bh)。
3)如果sf=0, 而of=1
of=1, 说明有溢出,逻辑上真正结果的正负≠实际结果的正负;
因sf=0,实际结果非负。而of=1说明有溢出,则结果非0,所以实际结果为正。
实际结果为正,而又有溢出,这说明是由于溢出导致了实际结果非负,简单分析一下,就可以看出,如果因为溢出导致了实际结果为正,那么逻辑上真正的结果必然为负。
这样:sf=0, of=1,说明了(ah)<(bh)。
4)如果sf=0,而of=0
of=0, 说明没有溢出,逻辑上真正结果的正负=实际结果的正负;
因sf=0,实际结果非负,所以逻辑上真正的结果非负,所以(ah)≥(bh)。