雇员数的存放有问题.你是这样写的
mov ax,ds:[si+0a8h]
.
.
.
add si,4
但是要注意,雇员数的大小是一个字,也就是两个字节
你给si加4,就是加了4个字节,这样,就会跳过一个数据。等到输入1986年的数据时,雇员数的值为零。因为你跳过了一半的数据,读取了未被定义的空间。
做除法的时候,除数为零,自然出现了除法溢出。
但是,对于你的程序,不能轻易的把add si,4修改为add si,2。因为你在输入年份和收入时使用了si寄存器,修改会造成更大的错误。
所以你得改变你的寻址方式。
建议用比较“傻瓜”的方式,比如:
start: mov ax,table
mov ds,ax
mov bx,0
mov ax,data
mov es,ax
mov bp,0
mov cx,21
mov di,0
s: mov si,2
mov ax,es:0[bp]
mov [bx],ax
MOV ax,es:2[bp]
mov [bx+2],ax
mov ax,es:84[bp]
mov [bx+5],ax
mov ax,es:84[bp][si]
mov [bx+7],ax
mov ax,es:168[di]
mov [bx+10],ax
mov dx,[bx+7]
mov ax,[bx+5]
div word ptr [bx+10]
mov [bx+13],ax
add bx,10h
add bp,4
add di,2
loop s
mov ax,4c00h
int 21h
codesg ends
你可以慢慢修改。
另外,学会自己调试程序,让别人帮忙修改程序的问题最好少提,因为太麻烦,很少有人会帮忙
assume cs:codesg,ss:stacksg
data segment
year db '1975','1976','1977','1978','1979','1980','1981','1982','1983'
db '1984','1985','1986','1987','1988','1989','1990','1991','1992'
db '1993','1994','1995'
gain dd 16,22,382,1356,2390,8000,16000,24486,50065,97479,140417,197514
dd 345980,590827,803530,1183000,1843000,2759000,3753000,4649000,5937000
;总收入
population 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
table segment
db 21 dup ('year summ ne ?? ')
table ends
stacksg segment stack
dw 0,0,0,0,0,0,0,0
stacksg ends
codesg segment
start: mov ax,data
mov ds,ax
mov ax,table
mov es,ax
mov ax,stacksg
mov ss,ax
mov sp,16
mov bx,0 ;bx存的是第几年
mov cx,21
s: push cx
mov si,bx ;si=year+bx*4 每个年份名称占4字节
mov cl,2
shl si,cl ;x4
add si,offset year; ds:si 指向该年的名字例如1975
mov di,bx ;di=bx*16 即第bx年对应的数组元素在table中的起始位置
mov cl,4
shl di,cl ;x16 因为table中每年占16字节
push di ;先留一下di一会还要复制人数
movsw ;把年份名称的4字节从ds:si复制到es:di
movsw
pop di
push di ;先留一下di一会还要放结果
add di,10 ;人数在table中是当年的16字节中的+10偏移
mov si,bx ;si=population +bx*4 每年人数值是dw,占4字节
shl si,1 ;x2
add si,offset population ; ds:si 指向该年的人数
movsw ;把人数复制到table里
pop di ;di是table中该年的记录位置
mov si,bx ;si=gain+bx*4 每年的收入是dd,占4字节
mov cl,2
shl si,cl ;x4
add si,offset gain; ds:si 指向该年的收入值
mov ax,[si]
mov dx,[si+2] ;dx:ax对中是该年的收入值
div word ptr es:[di+0ah] ;除以人数
mov es:[di+0dh],ax ;结果存入table
inc bx ;下一年
pop cx
loop s
mov ax,4c00h
int 21h
codesg ends
end start