MASM 在输入负小数时给出错误结果

问题描述

所以,我正在编写一个程序,将两个负整数相除并存储商和余数,但它显示的结果不正确。 这是我的代码

    mov eax,input1
    mov edx,0             ; dividend high half = 0.  prefer  xor edx,edx
    mov ebx,input2         ; divisor can be any register or memory
    div ebx
    mov remainder,edx       
    mov edx,offset prompt
    call Writestring
    mov quotient,eax 
    call WriteDec
    mov edx,offset prompt1
    call Writestring
    mov edx,remainder
    call WriteDec
    call Crlf

解决方法

div 用于无符号除法。对于带符号的除法,您需要 idiv。这需要 edx:eax 中的有符号 64 位被除数,因此如果除数为负,您不希望 edx 为零;相反,您想要 sign extend eax 进入 edx。这可以通过 cdq 完成。

所以我认为你想要

    mov eax,input1
    cdq 
    mov ebx,input2
    idiv ebx
    mov remainder,edx  
    mov quotient,eax
    ...

其他一些评论:

  • WriteDec 期望在哪里找到要写入的数字?检查其文档。您似乎没有将结果放入任何一致的寄存器中。

  • 您正在等待直到调用 Writestring 以将来自 eax 的商保存到内存中。您确定 Writestring 不会覆盖 eax 吗?许多调用约定都允许这样做。

  • 如果 input2 为 0,或者 input10x80000000(最负的 32 位整数)并且 {{1} } 是 input2。如果您在其他地方还没有它,您可能需要进行一些输入检查。

  • 正如您的评论所说,-1 可以从寄存器或内存中获取其操作数。因此,当您可以简单地执行 idiv 时,无需 mov ebx,input2。 (尽管您可能还想设计您的程序,使 idiv input2 始终存在于寄存器中,而从一开始就不需要存储在内存中。)