实验3 转移指令跳转原理及其简单应用编程

四、实验结论

1. 实验任务1

task1.asm源码:

assume cs:code, ds:data

data segment
    x db 1, 9, 3
    len1 equ $ - x

    y dw 1, 9, 3
    len2 equ $ - y
data ends

code segment
start:
    mov ax, data
    mov ds, ax

    mov si, offset x
    mov cx, len1
    mov ah, 2
 s1:mov dl, [si]
    or dl, 30h
    int 21h

    mov dl, ' '
    int 21h

    inc si
    loop s1

    mov ah, 2
    mov dl, 0ah
    int 21h

    mov si, offset y
    mov cx, len2/2
    mov ah, 2
 s2:mov dx, [si]
    or dl, 30h
    int 21h

    mov dl, ' '
    int 21h

    add si, 2
    loop s2

    mov ah, 4ch
    int 21h
code ends
end start

运行截图:

回答问题①:

 loop指令机器码后两位是F2,代表十进制数-14,可见其跳转的位移量是14。从CPU的角度来看,loop指令后第一个字节的偏移地址是001B,跳转的目的地址是000D,二者相减就得到了十进制数14。

回答问题②:

 loop指令机器码后两位是F0,代表十进制数-16,可见其跳转的位移量是16。从CPU的角度来看,loop指令后第一个字节的偏移地址是0039,跳转的目的地址是0029,二者相减就得到了十进制数16。

问题③:

 

2. 实验任务2

task2.asm源码:

assume cs:code, ds:data

data segment
    dw 200h, 0h, 230h, 0h
data ends

stack segment
    db 16 dup(0)
stack ends

code segment
start:  
    mov ax, data
    mov ds, ax

    mov word ptr ds:[0], offset s1
    mov word ptr ds:[2], offset s2
    mov ds:[4], cs

    mov ax, stack
    mov ss, ax
    mov sp, 16

    call word ptr ds:[0]
s1: pop ax

    call dword ptr ds:[2]
s2: pop bx
    pop cx

    mov ah, 4ch
    int 21h
code ends
end start

给出分析、调试、验证后,寄存器(ax) = ? (bx) = ? (cx) = ? 附上调试结果界面截图:

① 根据call指令的跳转原理,先从理论上分析,程序执行到退出(line31)之前,寄存器(ax) =指令 pop ax的偏移地址,寄存器 (bx) = 指令pop bx的偏移地址,寄存器(cx) = 当前代码的段地址。

② 对源程序进行汇编、链接,得到可执行程序task2.exe。使用debug调试,观察、验证调试结果与理论分析结果是否一致。

可看出ax的值是0021,是指令pop ax的偏移地址;bx的值是0026,是指令pop bx的偏移地址;cx的值是076c,是当前代码的段地址。与分析结果一致。

 

3. 实验任务3

给出程序源码task3.asm:

assume cs:code,ds:data
data segment
    x db 99, 72, 85, 63, 89, 97, 55
    len equ $ - x
data ends
code segment
start:
    mov ax,data
    mov ds,ax
    mov si,offset x
    mov cx,len
s:  mov al,[si]
    mov ah,0
    call printNumber
    call printSpace
    inc si    
    loop s
    mov ah,4ch
    int 21h
printNumber:
    mov bl,10
    div bl
    mov dl,al
    mov bh,ah
    add dl,30h
    mov ah,2
    int 21h
    mov dl,bh
    add dl,30h
    mov ah,2
    int 21h
    ret
printSpace:
    mov dl,20h
    mov ah,2
    int 21h
    ret
code ends
end start

运行测试截图:

 

 

4. 实验任务4

给出程序源码task4.asm:

assume cs:code,ds:data
data segment
    str db 'try'
    len equ $ - str
data ends
code segment
start:
    mov ax,data
    mov ds,ax
    mov ax,0b800h
    mov es,ax
    mov si,offset str        
    mov bl,2h
    mov bh,0
    mov di,0
    mov cx,len
    call printStr
    mov si,offset str
    mov bl,4h
    mov bh,25
    mov di,0f00h
    mov cx,len
    call printStr
    mov ah,4ch
    int 21h
printStr:
  s:mov al,[si]
    mov ah,bl
    mov es:[di],ax
    inc si
    add di,2
    loop s
    ret
code ends
end start

运行测试截图:

 

 

5. 实验任务5

给出程序源码task5.asm:

assume cs:code,ds:data
data segment
    stu_no db '201983290098'
    len = $ - stu_no
data ends
code segment
start:
    mov ax,0b800h
    mov es,ax
    mov ax,data
    mov ds,ax
    mov si,offset stu_no
    mov di,0
    mov al,0
    mov ah,17h
    mov cx,780h
  s:mov es:[di],ax
    add di,2
    loop s
    mov al,2dh
    mov cx,22h
 s1:mov es:[di],ax
    add di,2
    loop s1
    mov cx,0ch
 s2:mov al,[si]
    mov es:[di],ax
    add di,2
    inc si
    loop s2
    mov al,2dh
    mov cx,22h
 s3:mov es:[di],ax
    add di,2
    loop s3
    mov ah,4ch
    int 21h
 code ends
 end start

运行测试截图:

 

 

 五、实验总结

1.我熟悉了两种输出字符的方法。使用mov ah,2; int 21h的方式输出只能将字符输出到当前命令行的下一行;而将字符移动到显存空间,通过相应的寻址方式可以实现将字符显示在页面的任意位置。

2.我熟悉了div命令的使用方法和数据存放位置:对于8位除法,被除数放在ax中,余数和商放在ah和al中;也熟悉了call和ret指令的原理:call指令先把当前的代码段偏移地址IP压入栈中保存,再跳转到指定的地址,将该地址赋给IP。ret命令则将栈中保存的地址赋给IP。

posted @ 2021-11-27 18:22  Romantic2021  阅读(12)  评论(2)    收藏  举报