汇编语言(王爽)实验十

实验十

编写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

在这里插入图片描述

  • 23
    点赞
  • 92
    收藏
    觉得还不错? 一键收藏
  • 7
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 7
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值