实验七疑问
assume cs: codesgdata 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
;以上是表示21年公司雇员人数的21个word型数据
data ends
table segment
db 21 dup ('year summ ne ?? ')
table ends
stacksg segment
dw 0
stacksg ends
codesg segment
start:
mov ax,data
mov es,ax
mov ax,table
mov ds,ax
mov ax,stacksg
mov ss,ax
mov sp,2
mov bx,0
mov bp,0
mov di,0
mov cx,21;21行
s: push cx
mov ax,20h
;空格
mov ds:.4h,ax
mov ds:.9h,ax
mov ds:.0ch,ax
mov ds:.0fh,ax
mov si,0
mov cx,4;
s1: mov al,es:;获取年份
mov ds:,al;写入年份
mov al,es:.54h;获取公司总收入
mov ds:.5,al;写入公司总收入
inc si
loop s1
mov ax,es:.0a8h;获取雇员人数
mov ds:.0ah,ax;写入雇员人数
add di,2
mov ax,ds:.5;获取公司总收入
mov dx,ds:.7;
div word ptr ds:.0ah;公司总收入/雇员人数
mov ds:.0dh,ax;写入人均收入
add bx,4;存放下一个年份和公司总收入的位置
add bp,16;写入位置的下一行
pop cx
loop s
mov ax,4c00h
int 21h
codesg ends
end start
此代码功能是实现了,但结束不了,在最后cx等于0的时候不应该再会跳转到s标号处,但用debug调试的时候却会跳转到s标号处,导致会多执行一遍代码,其中进行除法时就是取除数时会取21个雇员人数后的那个数据,为0,除法就会报错,程序就结束不了 你好,鱼油!在你的代码中,CX循环计数器在最后变成0但依旧跳回`s`标号是由于PUSH和POP操作对堆栈的影响。你在`s`标号的开始处使用PUSH CX,将循环计数器CX保存到堆栈中,但你在结束时POP CX时把保存的计数器值恢复,使得CX不为0。从而导致循环继续进行。因此,为了解决这个问题,你需要将PUSH和POP操作与CX赋值区分开来。
修改如下:
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'
dd 16,22,382,1356,2390,8000,16000,24486,50065,97479,140417,197514
dd 345980,590827 , 803530,1183000,1843000,2759000,3753000,4649000,5937000
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
dw 0
stacksg ends
codesg segment
start:
mov ax, data
mov es, ax
mov ax, table
mov ds, ax
mov ax, stacksg
mov ss, ax
mov sp, 2
mov bx, 0
mov bp, 0
mov di, 0
mov cx, 21; 21行
s:
; 空格初始化
mov ax, 20h
mov ds:, ax
mov ds:, ax
mov ds:, ax
mov ds:, ax
mov si, 0
mov cx, 4
s1:
; 获取年份
mov al, es:
mov ds:, al
; 获取公司总收入
mov al, es:
mov ds:, al
inc si
loop s1
; 获取雇员人数
mov ax, es:
mov ds:, ax
add di, 2
; 公司总收入/雇员人数
mov ax, ds:
mov dx, ds:
div word ptr ds:
mov ds:, ax
; 更新位置
add bx, 4
add bp, 16
dec cx
jnz s
mov ax, 4C00h
int 21h
codesg ends
end start
这样的修改确保循环在CX==1时依旧会执行一次,但在这之后不再跳转到s标号,从而避免多执行一次代码。希望这样可以解决你的问题!如果还有其他疑问,请继续提问。
以上回复来自 FishC AI 小助理,如未能正确解答您的问题,请继续追问。 本帖最后由 jackz007 于 2024-9-26 17:00 编辑
请试试这个代码
assume cs: code , ds:data , es:table
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
;以上是表示21年公司雇员人数的21个word型数据
data ends
table segment
db 21 dup ('year summ ne ?? ')
table ends
stack segment stack
db 400h dup(0)
stack ends
code segment
start:
mov ax,data
mov ds,ax
mov ax,table
mov es,ax
mov bx,21*8
xor si,si
xor di,di
mov cx,21
cld
s: push cx
push si
movsw ; 传输年份字符串的头 2 个字符
movsw ; 传输年份字符串的后 2 个字符
inc di ; 跳过表格中的空格
add si,20*4
movsw ; 传输年度的总收入的低位 Word
movsw ; 传输年度的总收入的高位 Word
inc di ; 跳过表格中的空格
mov si,bx
movsw ; 传输年度的雇员人数
inc di ; 跳过表格中的空格
mov ax, es: ; 取得年度总收入的低位 Word
mov dx, es: ; 取得年度总收入的高位 Word
div word ptr es: ; 计算年度人均收入
stosw ; 保存年度人均收入
inc di ; 跳过表格中的空格
pop si
add si,4
add bx,2
pop cx
loop s
mov ax,4c00h
int 21h
code ends
end start
页:
[1]