gpt4 book ai didi

assembly - ASM 打印大数字

转载 作者:行者123 更新时间:2023-12-01 10:50:54 25 4
gpt4 key购买 nike

我应该在 NASM 中编写程序(并在 DosBox 下测试它),该程序将计算具有一个约束的阶乘:结果将存储在最多 128 位中。因此要计算的最大值是 factorial(34)。我的问题是我不知道如何打印这么大的数字。我唯一的提示是使用 DOS 中断,但经过长时间的研究后,我没有发现任何对我有帮助的东西。

我已经完成的计算阶乘的部分是:

        org 100h
section .text
factorial dw 1200h

xor edx, edx
xor eax, eax
mov ax, 6d ;we'll be calculating 6!
mov word [factorial+16], 1d ;current result is 1

wloop:
mov cx, 08h ;set loop iterator to 8
xor edx, edx ;clear edx
mloop: ;iterate over cx (shift)
mov bx, cx ;copy loop iterator to bx (indirect adressing will be used)
add bx, bx ;bx*=2
push ax ;store ax (number to multiply by)
mul word[factorial+bx] ;multiply ax and result
mov word[factorial+bx], ax ;store new result in [factorial+2*cx]
pop ax ;restore previous ax
push dx ;transfer from mul is stored in stack
loop mloop ;loop over cx until it's 0

mov cx, 7h ;set loop iterator to 7
tloop: ;iterate over cx, adding transfers to result
pop dx ;pop it from stack
mov bx, cx ;bx = cx
add bx, bx ;bx = bx+bx
adc [factorial+bx],dx ;add transfer to [factorial+2*cx]
loop tloop ;loop over cx until it's 0

pop bx ;one redundant transfer is removed from stack
dec ax ;decrease ax (number to multiply by)
cmp ax, 0 ;if ax is non-zero...
jne wloop ;...continue multiplying

movzx eax, word[factorial+16] ;load last 32 bits of result...
call println ;...and print them

jmp exit ;exit program

好吧...我知道如何打印 32 位数字,但这可能不是我想要打印 128 位数字的方式:

println:                    ;Prints from eax
push eax
push edx
mov ecx, 10
loopr:
xor edx, edx
div ecx ; eax <- eax/10, edx <- eax % 10
push eax ; stack <- eax (store, because DOS will need it)
add dl, '0' ; edx to ASCII
mov ah,2 ; DOS char print
int 21h ; interrupt to DOS
pop eax ; stack -> eax (restoring)
cmp eax, 0
jnz loopr

mov dl, 13d ;Carriage Return
mov ah, 2h
int 21h
mov dl, 10d ;Line Feed
mov ah, 2h
int 21h

pop edx
pop eax
ret

任何人都可以帮助我并给出一些如何处理的提示吗?我什至不知道阶乘是否计算正确,因为我无法打印大于 32b 的任何内容...

最佳答案

您的处理方式与处理小数的方式完全相同,即连续除以 10 并得到最小的数字作为余数。不要忘记在打印时颠倒数字的顺序(堆栈在这方面派上用场。)

通过从最高有效位到最低有效位遍历扩展整数中的单词,除以十,并将中间余数作为高位单词(在 DX 中,其中DIV 除 DX:AX。)

供引用的是 C 中的等效算法,您可以将其用作起点。其中 128 位数字存储为八个 little-endian 16 位字。

uint16_t extended_div(uint16_t *num, uint16_t den, size_t len) {
ldiv_t acc = { 0 };
num += len;
do {
acc = ldiv(acc.rem << 16 | *--num, den);
*num = acc.quot;
} while(--len);
return acc.rem;
}

为了获得额外的分数,您可以使用 x86 的 BCD 指令(在本例中为 AAM)作为替代方法,并直接以十进制格式计算阶乘的乘法。


首先,如果您还没有一个可用的调试器,请确保自己拥有一个可用的调试器,这样您就可以逐步完成测试用例并在纸上进行操作。就我个人而言,我以前使用过 Turbo Debugger,我相信 NASM 能够生成兼容的符号信息。我并不是要唠叨,但以前与 16 位 DOS 汇编有关的海报往往会忽略该过程的这一关键步骤。

关于assembly - ASM 打印大数字,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/20659282/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com