a撒野 发表于 2024-9-25 20:23:36

实验七疑问

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
;以上是表示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,除法就会报错,程序就结束不了

FishC 发表于 2024-9-25 20:23:57

你好,鱼油!在你的代码中,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 16:41:10

本帖最后由 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]
查看完整版本: 实验七疑问