实验3 转移指令跳转原理及其简单应用编程
实验任务1
源代码:
assume cs:code, ds:data data segment x db 1, 9, 3 len1 equ $ - x ; 符号常量, $指下一个数据项的偏移地址,这个示例中,是3 y dw 1, 9, 3 len2 equ $ - y ; 符号常量, $指下一个数据项的偏移地址,这个示例中,是9 data ends code segment start: mov ax, data mov ds, ax mov si, offset x ; 取符号x对应的偏移地址0 -> si mov cx, len1 ; 从符号x开始的连续字节数据项个数 -> cx 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 ; 取符号y对应的偏移地址3 -> si mov cx, len2/2 ; 从符号y开始的连续字数据项个数 -> cx 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

运行截图:

问题1:
line27, 汇编指令 loop s1 跳转时,是根据位移量跳转的。通过debug反汇编,查看其机 器码,分析其跳转的位移量是多少?(位移量数值以十进制数值回答)从CPU的角度,说明 是如何计算得到跳转后标号s1其后指令的偏移地址的。
答:loop s1的机器码为E2F2。跳转的位移量为-14。
loop指令后的第一个字节的偏移地址为:001B,而标号s1处的偏移地址为:000D,位移量=13(000D)-27(001B)=-14(补码表示为F2)。所以其跳转的位移量是-14。
问题2:
line44,汇编指令 loop s2 跳转时,是根据位移量跳转的。通过debug反汇编,查看其机 器码,分析其跳转的位移量是多少?(位移量数值以十进制数值回答)从CPU的角度,说明 是如何计算得到跳转后标号s2其后指令的偏移地址的。
答:loop s2的机器码为E2F0。跳转的位移量为-16。
Loop指令后的第一个字节的偏移地址为:0039,而标号s2处的偏移地址为:0029,位移量=41(0029)-57(0039)=-16(补码表示为F0)。所以其跳转的位移量为-16。
问题3:
反汇编:


实验任务2
源代码:
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
问题1:
根据call指令的跳转原理,先从理论上分析,程序执行到退出(line31)之前,寄存器(ax) = ? 寄存器(bx) = ? 寄存器(cx) = ?
答:ax=0021H,bx=0026H,cx=076CH.
理论分析:
Line13-line18:表示将s1的偏移地址,s2的偏移地址和cs即当前段地址都放入data段中,每个占2个字节,即1个字。
Line20-lin22:表示将stack段设置为栈段。
Line24:表示先将当前IP进栈,且当前IP为s1处的偏移地址,之后进行段内转移,执行后,IP=ds:[0],即s1的偏移地址0021,跳转到转移后的IP处,即执行s1,将当前栈中的元素出栈,而当前栈中即为s1的偏移地址,所以ax=0021H。
Line27:由于dword表示的是双字,所以该条指令表示为,先将当前CS进栈,再将当前IP进栈,当前CS和IP即为s2处的CS(076C)与IP(0026),之后再进行段间转移,执行后,CS=ds:[4],即076C,IP=ds:[2],即s2处的偏移地址0026,跳转到该位置后,即执行s2,将当前栈中元素出栈,即bx=0026。
Line29:将当前栈中元素出栈,即cx=076C。
问题2:
对源程序进行汇编、链接,得到可执行程序task2.exe。使用debug调试,观察、验证调试 结果与理论分析结果是否一致。




结果与理论一致。
实验任务3
源代码:
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,0
mov cx,7
s0:
call printNumber
call printSpace
loop s0
mov ax,4c00h
int 21h
printNumber:
mov al,[si]
mov ah,0
mov bl,10
div bl
mov dl,al
mov dh,ah
mov ah,2
or dl,30h
int 21h
mov dl,dh
or dl,30h
int 21h
inc si
ret
printSpace:
mov ah,2
mov dl,' '
int 21h
ret
code ends
end starts

![]()
实验任务4
源代码:
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 bh,0
mov bl,2h
call printStr
mov bh,24
mov bl,4h
call printStr
mov ax,4c00h
int 21h
printStr:
mov si,0
mov ah,0
mov al,bh
mov dx,160
mul dx
mov di,ax
mov cx,3
s0:
mov al,[si]
mov es:[di],al
mov es:[di+1],bl
add di,2
inc si
loop s0
ret
code ends
end starts
实验结果:


实验任务5
源代码:
assume cs:code
data segment
stu_no db '201983290171'
len = $ - stu_no
data ends
code segment
start:
mov ax,data
mov ds,ax
mov ax,0b800h
mov es,ax
call printBackground
mov bh,24
mov al,160
mul bh
mov bx,ax
call printChar
call printNum
call printChar
mov ax,4c00h
int 21h
printBackground:
mov si,1
mov bl,17h
mov cx,2000
s0:
mov es:[si],bl
add si,2
loop s0
ret
printChar:
mov al,'-'
mov cx,34
s1:
mov es:[bx],al
add bx,2
loop s1
ret
printNum:
mov di,0
mov cx,12
s2:
mov al,ds:[di]
mov es:[bx],al
add bx,2
inc di
loop s2
ret
code ends
end start
实验结果:


实验总结
通过本次实验,我对于书上第九章转移指令的原理和第十章CALL和RET指令有了更加深刻的理解和认识,在做关于第九章的实验内容中,我加深了操作符offset,jmp指令的多种执行方式,以及loop指令的多种执行方式的理解和操作,而第十章的内容,更是让我充分认识到call和ret指令联合使用时,可以使程序变得更加易懂和方便,也学会了mul指令的使用方法。而通过学习书本上实验9根据材料编程,也是让我懂得了如何在屏幕的各个地方显示数据,以及改变字体的属性等操作。
通过本次实验,也让我带着再次复习了以前所学习的相关知识,实在是使我受益良多。
浙公网安备 33010602011771号