王爽 著《汇编语言》(第3版):课程设计1

本文介绍了一个16位汇编程序设计案例,详细展示了数据段、代码段的组织方式,通过具体指令实现了数据的读取、计算及显示功能。程序计算了不同年份的人均收入,并将其结果显示在屏幕上,涉及数据处理、字符串转换和屏幕输出等关键操作。
摘要由CSDN通过智能技术生成
assume cs:code
data segment
	;第一段数据,偏移地址为0,本段一共4*21=84字节
	db '1975','1976','1977','1978','1979','1980','1981','1982','1983'
	db '1984','1985','1986','1987','1988','1989','1990','1991','1992'
	db '1993','1994','1995'
	
	;第二段数据,偏移地址为54H(前面有84字节),本段一共4*21=84字节
	dd 16,22,382,1356,2390,800,16000,24486,50065,97479,140417,197514
	dd 345980,590827,803530,1183000,1843000,2759000,3753000,4649000,593700
	
	;第三段数据,偏移地址为A8H(前面一共84+84字节)
	dw 3,7,9,13,28,38,130,220,476,778,1001,1442,2258,2793,4037,5635,8226
	dw 11542,14430,15257,17800
data ends

;人均收入计算结果
res segment
	dw 21 dup (0)
res ends

;32位除法计算结果临时存放空间
divtemp segment
	dw 8 dup (0)
divtemp ends

;处理显示字符串临时存放空间
strtemp segment
	db 16 dup (0)
strtemp ends

code segment
start:	mov ax,data
		mov ds,ax
		
		call mycls	;清屏
		call calres	;计算人均收入
		call show	;显示数据
		
		mov ax,4C00H
		int 21H

;清屏
mycls:	mov ax,0B800H
		mov es,ax
		mov cx,25	;屏幕显示共25行
		mov bx,0
		
c_s1:	push cx		;临时保存行偏移
		mov cx,80	;每行80个字符
		mov di,0
		
c_s2:	mov al,20H	;显示空格
		mov ah,07H	;黑底白字
		mov es:[bx+di],ax
		add di,2
		loop c_s1

		add bx,0A0H	;每行160个字节
		pop cx		;取回临时保存的行偏移
		loop c_s2:
		ret

;计算人均收入
calres:	mov ax,res
		mov es,ax
		mov cx,21
		mov si,0
		mov di,0

		;读取第二段数据作为被除数,第三段数据为除数
		;进行32位除法运算
cals:	mov ax,ds:[54H+di]		;16位放ax
		mov dx,ds:[54H+2+di]	;16位放dx
		mov bx,ds:[0A8H+si]		;除数
		div bx
		mov es:[si],ax			;商ax存入结果段,余数dx不采用
		
		add di,4				;dd数据段每次偏移+4
		add si,2				;dw数据段每次偏移+2
		loop cals
		ret

;处理数据并显示到屏幕
show:	mov ax,0B800H
		mov es,ax

		;年份
		mov cx,21
		mov si,0
		mov bx,0

ys1:	push cx
		mov cx,4
		mov di,0
		
		;读取第一段数据
ys2:	mov al,ds:[si]			;db类型数据,直接显示
		mov ah,07H				;控制背景颜色和字色
		mov es:[bx+4+di],ax		;控制显示位置,每行开始偏移2个字符的位置
		inc si
		add di,2
		loop ys2
		
		add bx,0A0H				;换行偏移0A0H,160个字节
		pop cx
		loop ys1

		;收入
		mov cx,21
		mov si,0
		mov bx,0
		
		;读取第二段数据
ins1:	push cx
		push si					;字符串偏移
		push bx					;行偏移
		mov ax,ds:[54H+si]		;偏移54H,低16位数据
		mov dx,ds:[54H+2+si]	;16位数据
		call dtoc				;处理32位数据显示
		
		pop bx					;取回行偏移
		mov di,0
		mov si,0

ins2:	mov ax,strtemp			;存放显示数据的临时空间
		mov es,ax
		mov dl,es:[di]			;取出转化成字符的数据
		mov dh,07H
		inc di					;db类型偏移增加1
		mov cl,dl				;读取到00为结尾
		mov ch,0
		jcxz in_ok				;读取完毕执行
		
		mov ax,0B800H
		mov es,ax
		mov es:[bx+24+si],dx	;第二段数据显示偏移24字节	
		add si,2				;dw类型数据每次偏移增加2
		jmp ins2

in_ok:	add bx,0A0H			;换行偏移0A0H,160个字节
		pop si
		add si,4				;dd类型数据每次偏移增加4
		pop cx
		loop ins1

		call clear				;清一下strtemp中的数据

		;人数
		mov cx,21
		mov si,0
		mov bx,0
		
		;读取第三段数据
peos1:	push cx
		push si
		push bx
		
		mov ax,ds:[0A80H+si]	;偏移A80H
		mov dx,0				;处理32位数据显示,由于第三段数据是dw类型,所以高16位都是0
		call dtoc
		pop bx
		mov di,0
		mov si,0

peos2:	mov ax,strtemp
		mov es,ax
		mov dl,es:[di]
		mov dh,07H
		inc di
		mov cl,dl
		mov ch,0
		jcxz peo_ok
	
		mov ax,0B800H
		mov es,ax
		mov es:[bx+44+si],dx
		add si,2
		jmp peos2

peo_ok:	add bx,0A0H
		pop si
		add si,2
		pop cx
		loop peos1

		call clear
		
		;人均收入
		mov cx,21
		mov si,0
		mov bx,0
		mov ax,res		;读取计算结果段
		mov ds,ax
		
avers1:push cx
		push si
		push bx

		mov ax,ds:[si]
		mov dx,0
		call dtoc
		pop bx
		mov di,0
		mov si,0

avers2:	mov ax,strtemp
		mov es,ax
		mov dl,es:[di]
		mov dh,07H
		inc di
		mov cl,dl
		mov ch,0
		jcxz aver_ok
		
		mov ax,0B800H
		mov es,ax
		mov es:[bx+64+si],dx
		add si,2
		jmp avers2

aver_ok:add bx,0A0H
		pop si
		add si,2
		pop cx
		loop avers1
		ret

;处理显示数据
dtoc:	mov di,0
dts:	jcxz dt_ok
		;mov ax,ax
		;mov dx,dx
		mov cx,0AH
		call divdw

		add cx,30H		;显示ASCII字符加30H
		push cx
		inc di
		mov cx,ax
		jmp dts

dt_ok:	mov cx,di
		mov ax,strtemp
		mov es,ax
		mov di,0

dt_oks:	pop ax
		mov es:[di],al
		inc di
		loop dt_oks
		ret

;不会产生溢出的除法运算
;X/N=int(H/N)*65536+(rem(H/N)*65536+L)/N
;L被除数低16;H被除数高16位
divdw:	push ax		;被除数低16位,栈地址偏移[bp+6],L
		push dx		;被除数高16位,栈地址偏移[bp+4],H
		push cx		;除数,栈地址偏移[bp+2],N
		push bp
		mov bp,sp
		
		mov ax,divtemp
		mov es,ax
		
		;H/N进行32位除法运算,高16位放0
		mov ax,[bp+4];H
		mov dx,0
		mov bx,[bp+2];N
		div bx
		
		;商在ax,余数在dx
		;ax(dx)*65536相当于ax(dx)放高16位,低16位为0
		mov word ptr es:[0],0		;int(H/N)*65536的结果的低16位
		mov es:[2],ax				;int(H/N)*65536的结果的高16位
		mov word ptr es:[4],0		;rem(H/N)*65536的结果的低16位
		mov es:[6],dx				;rem(H/N)*65536的结果的的高16;rem(H/N)*65536的结果+L,rem(H/N)*65536的结果的低16+L
		;由于rem(H/N)*65536的结果的低16位为0,只需要mov L即可
		mov ax,[bp+6]
		mov es:[4],ax
		
		;计算(rem(H/N)*65536+L)/N,32位除法
		mov ax,es:[4]		;16位
		mov ds,es:[6]		;16位
		mov bx,[bp+2]		;N
		div bx
		;处理计算结果
		mob cx,dx;余数放cx
		add es:[0],ax		;商(16位)与int(H/N)*65536的结果相加
		mov dx,es:[2]		;最终结果的高16位放dx
		mov ax,es:[0]		;最终结果的低16位放ax
		pop bp
		pop bx
		pop bx
		pop bx				;清空传入栈的数据,有没有更好的办法?
		ret

;清空strtemp
clear:	mov ax,strtemp
		mov es,ax
		mov cx,16
		mov di,0
cles:	mov byte ptr es:[di],0
		inc di
		loop cles
		ret

code ends
end start

问题:
1.显示数据的年份、收入、人数、人均四个段的代码类似,应该考虑用个通用的方法解决。循环?

2.栈中数据使用完后的处理方法。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值