汇编语言(王爽)实验十
程序员文章站
2024-03-23 22:45:28
...
实验十
编写3个子程序
1、显示字符串
功能:在指定的位置,用指定的颜色,显示一个用0结束的字符串
参数:(dh)=行号、(dl)=列号、(cl)=颜色、ds:si指向字符串的首地址
行、列从0开始计数
如要求在屏幕的8行3列,用绿色显示data段中的字符串
assume cs:code
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: mov ax,0B800H ; 确定输入字符串的段地址
mov es,ax
mov al,00A0H ; 每行差00A0个字节,不能字母开头
mul dh ; 计算出的"行偏移地址"存在ax中
push ax ; 入栈,暂存
mov al,2 ; 每一列占两个字节,第2个字节
mul dl ; 计算出的"列偏移地址"存在ax中
pop di ; 取出之前的"行偏移"
add di,ax ; "行偏移"+"列偏移"=总的偏移地址
input: push cx ; jczx要用到cx,会影响颜色的输入
mov cx,[si] ; 从data中取字符
jcxz end_input ; 若取到末尾,跳转
mov es:[di],cx
inc si ; 指向下一个字符
inc di
pop cx
mov es:[di],cl ; 存入颜色信息
inc di
jmp short input
end_input: pop cx ; 在到了字符串末尾之后,
; 之前的cx还在栈中,要出栈
; 否则ret从栈中取出原来的IP值时会出错
ret
code ends
end start
2. 解决除法溢出问题
功能:进行不会产生溢出的除法运算,被除数为dword,除数为word,结果为dword
参数:(ax)=dword型数据的低16位
(dx)=dword型数据的高16位
(cx)=除数
返回:(dx)=结果的高16位
(ax)=结果的低16位
(cx)=余数
思路:对于32位的被除数,先高位的16位在前面补0组成一个32位数除以除数,得到的商便是最终结果的高16位,而得到的余数作为一个新的32位数的高16位,而被除数的低16位作为新32位数中的低16位,除以被除数,得到的余数就是最终的余数,而得到的商就是结果的低16位
assume cs:code
code segment
start: mov ax,4240H
mov dx,000FH
mov cx,0AH
call divdw
mov ax,4c00h
int 21h
divdw: push ax ; 先高位除以被除数,暂存ax
mov ax,dx
mov dx,0 ; 除数是16位,被除数应该为32位,高位存储在dx中
div cx
mov bx,ax ; 用bx存储ax,即商
pop ax ; 弹出低位的值
div cx ; 此时,第一次除法运算得到的余数作为高16位的值
; 存储在在dx中
mov cx,dx
mov dx,bx
ret
code ends
end start
3.数值显示
把数据如12666以十进制显示在屏幕上
思路:12666每次除以10取余,数字加30H就是对应的字符的ASCII码值(字符“0”~“9”对应的ASCII码是30H-39H),将数据转换为字符串,然后调用show_str显示在屏幕上
assume cs:code
data segment
db 10 dup (0)
data ends
code segment
start: mov ax,12666
mov bx,data
mov ds,bx
mov si,0
call dtoc
mov dh,8
mov dl,3
mov cl,2
call show_str
mov ax,4c00h
int 21h
dtoc: mov bx,10 ; 除数
mov dx,0 ; dx为32位被除数的高16位
div bx
inc si ; 记录数的长度
mov cx,ax ; 把商赋值给cx,用于jcxz
push dx ; 判断前先把余数入栈
jcxz ok
jmp short dtoc
ok: mov cx,si ; 数的长度就是循环的次数
mov si,0 ; si为data段的偏移地址
s: pop bx
add bx,30H ; 数字的大小+30H就是对应字符的
; ASCII码
mov [si],bl ; 存储那些ASCII码只需一个字节
inc si
loop s
mov [si],cl ; 由于show_str使用时
; 默认字符串以0结尾,因此在最后补0
mov si,0 ; 修改为0保证show_str的正确执行
ret
show_str: mov ax,0B800H ; 确定输入字符串的段地址
mov es,ax
mov al,00A0H ; 每行差00A0个字节,不能字母开头
mul dh ; 计算出的"行偏移地址"存在ax中
push ax ; 入栈,暂存
mov al,2 ; 每一列占两个字节,第2个字节
mul dl ; 计算出的"列偏移地址"存在ax中
pop di ; 取出之前的"行偏移"
add di,ax ; "行偏移"+"列偏移"=总的偏移地址
input: push cx ; jczx要用到cx,会影响颜色的输入
mov cx,[si] ; 从data中取字符
jcxz end_input ; 若取到末尾,跳转
mov es:[di],cx
inc si ; 指向下一个字符
inc di
pop cx
mov es:[di],cl ; 存入颜色信息
inc di
jmp short input
end_input: pop cx ; 在到了字符串末尾之后,
; 之前的cx还在栈中,要出栈
; 否则ret从栈中取出原来的IP值时会出错
ret
code ends
end start
上一篇: Java第一步Hello World