优化 ok1:mov cx,dx jcxz ok2 ok2:pop ax pop dx mov cx,di mov ax,di mov dx,7 sub dx,ax s6:pop ax add al,30h mov es:[si],al inc si mov al,02h mov es:[si],al inc si loop s6 mov cx,dx s7: inc si inc si loop s7 ret
时间: 2024-02-10 10:17:04 浏览: 26
这段代码的作用是将 DX 和 DI 中的值组成的数转换为十进制数,并存放到以 SI 指向的内存位置中。优化的话可以考虑减少循环次数,从而提高效率。可以使用除法运算逐位计算十进制数的每一位,然后将其转换为 ASCII 码并存储到内存中。可以改写为:
ok1:
mov cx,dx
jcxz ok2
mov bx,10 ; 将被除数设置为 10
s6:
xor dx,dx ; DX 存放余数
div bx ; AX 存放商
add dl,30h ; 转换为 ASCII 码
mov es:[si],dl ; 存放到内存中
inc si
cmp ax,0
jne s6
mov cx,di
mov ax,di
mov dx,7
sub dx,ax
s7:
mov al,02h
mov es:[si],al
inc si
loop s7
ret
其中,DIV 指令用于将 DX:AX 中的值除以 BX,并将商存放在 AX 中,余数存放在 DX 中。每次计算出一个余数,就将其转换为 ASCII 码并存储到内存中。循环直到商为 0。最后将 DI 和 DX 中的值转换为 ASCII 码并存储到内存中,中间插入一个字符 02h 作为分隔符。
相关问题
完善 s8: inc si inc si loop s8;补上4个空格 mov ah,2 mov al,bl mul ah add ax,168 mov bp,ax mov ax,ds:[bp] inc bp inc bp mov dx,0000 mov cx,10 push ax call calcul pop cx pop ax pop dx call divdw call calcul pop cx inc bl;恢复循环次数and行号加一 sub cx,1 jcxz ok jmp s
s8: inc si inc si loop s8 ; mov ah,2 mov al,bl mul ah add ax,168 mov bp,ax mov ax,ds:[bp] inc bp inc bp mov dx,0000 mov cx,10 push ax call calcul pop cx pop ax pop dx call divdw call calcul pop cx inc bl ; 恢复循环次数and行号加一 sub cx,1 jcxz ok jmp s
完善assume cs:code,ds:data data segment db 'welcome to masm!',0 data ends code segment start:mov dh,8 mov dl,3 mov cl,2 mov ax,data mov ds,ax mov si,0 call show_str mov ax,4c00h int 21h show_str: push ax push bx push es push si push cx mov ax,0b800h mov es,ax mov ax,160 mul dh mov bx,ax mov ax,2 mul dl add bx,ax mov al,cl mov cl,0 show0: mov ch,[si] jcxz show1 mov es:[bx],ch mov es:[bx].1,al inc si add bx,2 jmp show0 show1: pop cx pop si pop es pop bx pop ax ret code ends end start使其能在屏幕中间分别显示绿色、绿底红色、白底蓝色的‘welcome to masm’。
在原有代码的基础上,我们需要对 `show_str` 进行修改,以便可以在屏幕中间分别显示绿色、绿底红色、白底蓝色的字符串。
修改后的代码如下:
```
show_str:
push ax
push bx
push es
push si
push cx
; 显示绿色字符串
mov ax, 0B800h
mov es, ax
mov ax, 160
mul dh
mov bx, ax
mov ax, 2
mul dl
add bx, ax
mov al, 2 ; 前景色为绿色
mov cl, 0 ; 背景色为黑色
mov si, offset data
call show_line
; 显示绿底红色字符串
mov ax, 0B800h
mov es, ax
mov ax, 160
mul dh
mov bx, ax
mov ax, 2
mul dl
add bx, ax
mov al, 4 ; 前景色为红色,背景色为绿色
mov cl, 2
mov si, offset data
call show_line
; 显示白底蓝色字符串
mov ax, 0B800h
mov es, ax
mov ax, 160
mul dh
mov bx, ax
mov ax, 2
mul dl
add bx, ax
mov al, 1 ; 前景色为蓝色,背景色为白色
mov cl, 15
mov si, offset data
call show_line
pop cx
pop si
pop es
pop bx
pop ax
ret
show_line:
push ax
push bx
; 计算字符串长度
mov cx, 0
mov bx, si
calc_length:
mov al, [si]
cmp al, 0
je done_calc_length
inc cx
inc si
jmp calc_length
done_calc_length:
mov si, bx
; 计算偏移量
mov bx, 0
mov ax, 160
mul dh
add bx, ax
mov ax, 2
mul dl
add bx, ax
mov ax, cx
sub ax, 1
shr ax, 1
mov dx, 80
sub dx, ax
mov bx, [bx]
add bx, dx
; 显示字符串
mov si, offset data
show_char:
lodsb
cmp al, 0
je done_show_line
mov ah, cl
stosw
jmp show_char
done_show_line:
pop bx
pop ax
ret
```
在 `show_str` 中,我们先使用 `show_line` 函数分别显示绿色、绿底红色、白底蓝色的字符串。这里我们将计算字符串长度和偏移量的部分提取出来,单独写了一个 `show_line` 函数。在 `show_line` 中,我们先计算字符串长度,然后计算偏移量,最后在屏幕上显示字符串。
需要注意的是,我们在 `show_str` 中调用了三次 `show_line` 函数,每次需要设置不同的前景色和背景色。在 `show_line` 中,我们使用 `ah` 存储前景色,使用 `cl` 存储背景色。在显示绿色字符串时,我们将 `al` 设置为 2,表示前景色为绿色,背景色为黑色。在显示绿底红色字符串时,我们将 `al` 设置为 4,表示前景色为红色,背景色为绿色。在显示白底蓝色字符串时,我们将 `al` 设置为 1,表示前景色为蓝色,背景色为白色。
完整的代码如下:
```
assume cs:code, ds:data
data segment
db 'welcome to masm!',0
data ends
code segment
start:
mov dh, 8
mov dl, 3
mov cl, 2
mov ax, data
mov ds, ax
mov si, 0
call show_str
mov ax, 4c00h
int 21h
show_str:
push ax
push bx
push es
push si
push cx
; 显示绿色字符串
mov ax, 0B800h
mov es, ax
mov ax, 160
mul dh
mov bx, ax
mov ax, 2
mul dl
add bx, ax
mov al, 2 ; 前景色为绿色
mov cl, 0 ; 背景色为黑色
mov si, offset data
call show_line
; 显示绿底红色字符串
mov ax, 0B800h
mov es, ax
mov ax, 160
mul dh
mov bx, ax
mov ax, 2
mul dl
add bx, ax
mov al, 4 ; 前景色为红色,背景色为绿色
mov cl, 2
mov si, offset data
call show_line
; 显示白底蓝色字符串
mov ax, 0B800h
mov es, ax
mov ax, 160
mul dh
mov bx, ax
mov ax, 2
mul dl
add bx, ax
mov al, 1 ; 前景色为蓝色,背景色为白色
mov cl, 15
mov si, offset data
call show_line
pop cx
pop si
pop es
pop bx
pop ax
ret
show_line:
push ax
push bx
; 计算字符串长度
mov cx, 0
mov bx, si
calc_length:
mov al, [si]
cmp al, 0
je done_calc_length
inc cx
inc si
jmp calc_length
done_calc_length:
mov si, bx
; 计算偏移量
mov bx, 0
mov ax, 160
mul dh
add bx, ax
mov ax, 2
mul dl
add bx, ax
mov ax, cx
sub ax, 1
shr ax, 1
mov dx, 80
sub dx, ax
mov bx, [bx]
add bx, dx
; 显示字符串
mov si, offset data
show_char:
lodsb
cmp al, 0
je done_show_line
mov ah, cl
stosw
jmp show_char
done_show_line:
pop bx
pop ax
ret
code ends
end start
```