汇编语言实验四
实验任务1
验证性实验:有些汇编指令会影响到标志寄存器中的一个或多个状态标志位。
源代码:
assume cs:code, ds:data
data segment
x dw 1020h, 2240h, 9522h, 5060h, 3359h, 6652h, 2530h, 7031h
y dw 3210h, 5510h, 6066h, 5121h, 8801h, 6210h, 7119h, 3912h
data ends
code segment
start:
mov ax, data
mov ds, ax
mov si, offset x
mov di, offset y
call add128
mov ah, 4ch
int 21h
add128:
push ax
push cx
push si
push di
sub ax, ax
mov cx, 8
s: mov ax, [si]
adc ax, [di]
mov [si], ax
add si, 2
add di, 2
loop s
pop di
pop si
pop cx
pop ax
ret
code ends
end start
①观察结果可以发现
使用add命令,会进位,而且会置零
使用inc命令,不会进位,会置零
line31~line34的4条inc指令,能否替换成如下代码?你的结论的依据/理由是什么?
②是不可以更换的,因为inc换成add有可能引起进位,导致原本计算中的adc计算错误。
在debug中调试,观察数据段中做128位加之前,和,加之后,数据段的值的变化。
③不改变代码:
改变代码:
add si, 2
add di, 2
根据这个结果虽然看不出区别,但是根据上面的例子我们可以发现,inc换成add有可能引起进位,导致原本计算中的adc计算错误,所以是不能换的。
实验任务2
代码:
assume cs:code, ds:data
data segment
str db 80 dup(?)
data ends
code segment
start:
mov ax, data
mov ds, ax
mov si, 0
s1:
mov ah, 1
int 21h
mov [si], al
cmp al, '#'
je next
inc si
jmp s1
next:
mov ah, 2
mov dl, 0ah
int 21h
mov cx, si
mov si, 0
s2: mov ah, 2
mov dl, [si]
int 21h
inc si
loop s2
mov ah, 4ch
int 21h
code ends
end start
运行测试截图:
① 汇编指令代码line11-18,实现的功能是?
循环输入值同时输入的字符是否是#,如果是就跳转到next
② 汇编指令代码line20-22,实现的功能是?
换行
③ 汇编指令代码line24-30,实现的功能是?
循环输出
实验任务3
编写8086汇编源程序task3.asm,在屏幕上以十进制形式输出data段中这一组连续的数据,数据和数据之间以空格间隔。
代码:
assume cs:code, ds:data
data segment
x dw 91, 792, 8536, 65521, 2021
len equ $ - x
data ends
code segment
start:
mov ax, data
mov ds, ax
mov si, offset x
mov cx, 5
s: mov ax, [si]
push cx
call printNumber
call printSpace
add si, 2
pop cx
loop s
mov ah, 4ch
int 21h
printNumber:
mov dx, 0
mov cx, 0
s1: mov bx, 10
div bx
push dx
mov dx, 0
inc cx
add ax, 0
jnz s1
s2: pop dx
mov ah, 2
add dl, '0'
int 21h
loop s2
ret
printSpace:
mov ah, 2
mov dl, ' '
int 21h
ret
code ends
end start
测试结果:
实验任务4
编写8086汇编源程序task4.asm,将data段中字符串里的小写字符转换成大写。
代码:
assume cs:code, ds:data
data segment
str db "assembly language, it's not difficult but tedious"
len equ $ - str
data ends
code segment
start:
mov ax, data
mov ds, ax
mov si, offset str
mov cx, len
s: call strupr
mov ah, 4ch
int 21h
strupr:
s1: mov al, [si]
cmp al, 'a'
jb s2
cmp al, 'z'
ja s2
sub byte ptr [si], 32
s2: inc si
loop s1
code ends
end start
调试结果:
实验任务5
运行程序,输入7,观察结果。输入其他字符,观察结果。结合运行结果和注释,理解代码实现的功能。
源代码:
assume cs:code, ds:data
data segment
str1 db "yes", '$'
str2 db "no", '$'
data ends
code segment
start:
mov ax, data
mov ds, ax
mov ah, 1
int 21h
mov ah, 2
mov bh, 0
mov dh, 24
mov dl, 70
int 10h
cmp al, '7'
je s1
mov ah, 9
mov dx, offset str2
int 21h
jmp over
s1: mov ah, 9
mov dx, offset str1
int 21h
over:
mov ah, 4ch
int 21h
code ends
end start
测试结果:
理解:
首先设置输出光标在24行70列
然后使用1号子功能读入一个数字和7进行比较
不是7输出no,是7输出yes
实验任务6
通过此项实现任务,你对中断、软中断实现机制的理解
我的理解:
中断机制是计算机在遇到系统错误,或者int时产生软中断。
首先计算机收到中断信号,通过查软中断向量表获得对应中断代码的段地址和偏移量,然后执行相应位置的代码。
所以编写一个软中断指令的可以分为以下三步:
1.拷贝中断代码到系统空闲的内存处
2.在中断向量表中添加相应的中断代码的段地址和偏移量
3.编写中断代码
实验总结
- 本次实验综合了上述几次试验,强化了代码编写能力,加深了对flag标志寄存器的理解,从而可以更加熟练地使用cmp和adc等指令,也认识到了add和inc的区别
- 最后一个任务学到了可以通过offset计算代码的长度,不用自己数了。也对中断有了更深入的了解,学会了如何编写自己的软中断程序。