汇编语言 王爽 【第四版】 实验7 寻址方式在结构化数据访问中的应用

在线进制转换

方法一:一次遍历
首先把书本给的代码跑一下,查看各个数据的分布,注意psp占100H,ds=075A 而dataseg首地址应该是075a*16 +100H =076a

为什么要X16,涉及到实地址=段地址X16+段内偏移 的技术

我们d命令观察dataseg数据段在内存中的分布,发现表示21年的是字符串,其长度为21个X一个4B =84B =54h ,如图中的076A:0050 行第3个就是最后的年份’1995’

其后是10 00 按照内存大端存储,54H段内偏移位置表示的是0010H 对应存储的数据就是十进制的16!也就是公司总收入的第一个数据16

这里我们发现,只用字符串在debug里会显示其ASCII表示的字符,数值不会显示而会显示对应ascii码的字符

汇编语言 王爽 【第四版】 实验7 寻址方式在结构化数据访问中的应用_第1张图片
这样我们便知道了总收入数据的起始偏移地址,而年份起始偏移地址是0
那么雇员的呢?我们看总收入最后一个数值是5937000 =00 5a 97 68 ,在内存中存放就是 68 97 5a 00 显然不难发现其00最后在00a7,所以下一数据即雇员人数从00a8偏移地址开始!


我们不妨手推一下,年份是db,代表字符串每一个字符占1个B,所以一个‘1 9 7 5’占4B,总长度为21*4=84B =54h,而从0开始,所以下一段开头是54h

一开始这里理解错误,db ‘1975’=‘1’ ‘9’ ‘7’ ‘5’ 每个一个B,一共4B!!!

总收入是dd 双字 4B 跟年份一致(方法二这里可以一起处理),长度也是54h,54+54h=a8h ,又因为从0开始的,所以雇员开头就是a8h

至此,我们三个类型的数据的首地址均确定了,可以用一次大循环来处理即可

assume cs:codesg
 
data segment
	db '1975','1976','1977','1978','1979','1980','1981','1982','1983'
	db '1984','1985','1986','1987','1988','1989','1990','1991','1992'
	db '1993','1994','1995'
	;以上是表示21年的21个字符串
	
	dd 16,22,382,1356,2390,8000,16000,24486,50065,97479,140417,197514
	dd 345980,590827,803530,1183000,1843000,2759000,3753000,4649000,5937000
	;以上是表示21年公司总收入的21个dword型数据
	
	dw 3,7,9,13,28,38,130,220,476,778,1001,1442,2258,2793,4037,5635,8226
	dw 11542,14430,15257,17800
	;以上是表示21Ian公司雇员人数的21个word型数据	
data ends
 
table segment
	db 21 dup ('year summ ne ?? ')
table ends
;以上为书中给出的代码
 
stacksg segment
	dw 0,0,0,0,0,0,0,0
	;开辟栈空间,存放外层循环CX的值
stacksg ends
 
codesg segment
start:	
	
	mov ax,data
	mov ds,ax
	mov ax,table
	mov es,ax ;es附加寄存器,存放目标地址
	
	mov bx,0;用于原遍历列
	mov di,0;用于table一行写,目标
	mov si,0;源
	mov cx,21 ;次数21次,是10进制
	
s:	mov al,[bx];默认从ds:[bx]访问,即第一个年份开始的位置,因为是1B,所以用al
	mov es:[di],al
	mov al,[bx+1]
	mov es:[di+1],al
	mov al,[bx+2]
	mov es:[di+2],al
	mov al,[bx+3]
	mov es:[di+3],al;依次把年份'1' '9' '7' '5'写入,这边可以用一个loop其实
		
	mov ax,54h[bx] ;因为dd是双字,需要用两个寄存器dx高位 ax低位,这里等价于mov ax,[54h+bx]
	mov dx,56h[bx] ;54h是观察data区代表收入的偏移地址起始,低位2字,56h是高位2字
	mov es:5h[di],ax 
	mov es:7h[di],dx ;写入收入
	
	mov ax,0a8h[si];必须加0,因为在汇编里数据不能以字母开头
	mov es:0ah[di],ax;写入雇员数
	
	mov ax,54h[bx];就等价于 mov ax,ds:[54h+bx] ax存放被除数低16位,bx存放高16位
	div word ptr ds:0a8h[si];因为被除数是双字dd,除数是dw 因此结果是word
	mov es:0dh[di],ax;ax存放商,只要这一部分,就相当于取整了
	
	add si,2;用于单字,2B
	add bx,4;用于双字,4B
	add di,16;用于换行
	
	loop s
	
	
	mov ax,4c00h
	int 21h
codesg ends
end start

方法二:

注意 代码中数据加 h和不加h的区别,
不加h,表示立即数,B字节为单位,表示跨过多少字节如84
加h,表示偏移地址,如54h

你可能感兴趣的:(专业课学习,其他)