Распечатать целое число для консоли в сборке x86
Вопрос
Когда я добавляю два значения в 16-битной сборке, какой лучший способ распечатать результат к консоли?
На данный момент у меня есть этот код:
;;---CODE START---;;
mov ax, 1 ;put 1 into ax
add ax, 2 ; add 2 to ax current value
mov ah,2 ; 2 is the function number of output char in the DOS Services.
mov dl, ax ; DL takes the value.
int 21h ; calls DOS Services
mov ah,4Ch ; 4Ch is the function number for exit program in DOS Services.
int 21h ; function 4Ch doesn't care about anything in the registers.
;;---CODE END---;;
Я думаю, что значение DL должно быть в коде ASCII, но я не уверен, как преобразовать значение топора после добавления в ASCII.
Решение
Вы в основном хотите разделить на 10, распечатать остаток (одну цифру), а затем повторить с фактором.
; assume number is in eax
mov ecx, 10
loophere:
mov edx, 0
div ecx
; now eax <-- eax/10
; edx <-- eax % 10
; print edx
; this is one digit, which we have to convert to ASCII
; the print routine uses edx and eax, so let's push eax
; onto the stack. we clear edx at the beginning of the
; loop anyway, so we don't care if we much around with it
push eax
; convert dl to ascii
add dl, '0'
mov ah,2 ; 2 is the function number of output char in the DOS Services.
int 21h ; calls DOS Services
; now restore eax
pop eax
; if eax is zero, we can quit
cmp eax, 0
jnz loophere
В качестве бокового примечания у вас есть ошибка в вашем коде прямо здесь:
mov ax, 1 ;put 1 into ax
add ax, 2 ; add 2 to ax current value
mov ah,2 ; 2 is the function number of output char in the DOS Services.
mov dl, ax ; DL takes the value.
Вы положены 2
в ah
, а потом вы положили ax
в dl
. Отказ Вы в основном мусор ax
прежде чем печатать его.
У вас также есть несоответствие размера с момента dl
8 бит широко и ax
16 бит широко.
Что вы должны сделать, это перевернуть последние две строки и исправить несоответствие размера:
mov ax, 1 ;put 1 into ax
add ax, 2 ; add 2 to ax current value
mov dl, al ; DL takes the value.
mov ah,2 ; 2 is the function number of output char in the DOS Services.
Другие советы
Просто исправляю заказ кода @nathan Feedman's
PrintNumber proc
mov cx, 0
mov bx, 10
@@loophere:
mov dx, 0
div bx ;divide by ten
; now ax <-- ax/10
; dx <-- ax % 10
; print dx
; this is one digit, which we have to convert to ASCII
; the print routine uses dx and ax, so let's push ax
; onto the stack. we clear dx at the beginning of the
; loop anyway, so we don't care if we much around with it
push ax
add dl, '0' ;convert dl to ascii
pop ax ;restore ax
push dx ;digits are in reversed order, must use stack
inc cx ;remember how many digits we pushed to stack
cmp ax, 0 ;if ax is zero, we can quit
jnz @@loophere
;cx is already set
mov ah, 2 ;2 is the function number of output char in the DOS Services.
@@loophere2:
pop dx ;restore digits from last to first
int 21h ;calls DOS Services
loop @@loophere2
ret
PrintNumber endp
Основной алгоритм:
divide number x by 10, giving quotient q and remainder r
emit r
if q is not zero, set x = q and repeat
Обратите внимание, что это даст цифры в обратном порядке, поэтому вы, вероятно, захотите заменить шаг «Emit» с чем-то, что хранит каждую цифру, чтобы позже вы могли бы потратить в обратном направлении через сохраненные цифры.
Кроме того, обратите внимание, что для преобразования двоичного числа от 0 до 9 (десятичное число) до ASCII просто добавьте код ASCII для «0» (который 48) на номер.
mov dl, ax
Это не будет работать как dl
а также ax
имеют разные размеры битов. То, что вы хотите сделать, это создать цикл, в которой вы разделяете 16-битное значение на 10, запомните остальное в стеке, а затем продолжите цикл с результатом целочисленного подразделения. Когда вы достигаете результата 0, очистите цифру стека по цифре, добавив 48 к цифрам, чтобы превратить их в цифры ASCII, затем распечатайте их.