实验4 8086标志寄存器及中断
实验任务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
加之后:
可以看出成功相加,
line31~line34的4条inc指令,能否替换成如下代码?
add si, 2
add di, 2
在这个程序代码中,我们进行一次替换,观察结果有无变化。
可以看到修改之后,对于结果没有变化,这是为什么呢。
为了探究这个问题,我们换一种方式。
一次执行一下两个代码段:
mov al,ff
add al,1
mov al,ff
inc al
结果如下:
可以得到结论,add
指令对于标志寄存器中的零标志位ZF
有影响,对进位标志位CF
也有影响。
而inc
指令仅仅对ZF
零标志位有影响,对进位标志位CF
无影响。
所有在这个实验中用到了adc
指令,这是带进位的加法,因此若是用add
指令很有可能对结果有影响,而针对这个程序,由于相加的数据段恰巧未生成进位,所以对结果是没影响的。但是我们得到了普遍的结论就是在这种情况下不能用add
指令替换inc
指令。
实验任务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,实现的功能是?
实现字符的录入。
② 汇编指令代码line20-22,实现的功能是?
0ah的是换行的ascii码。
打印一个换行
③ 汇编指令代码line24-30,实现的功能是?
回答问题:
① 汇编指令代码line11-18,实现的功能是?
实现字符的录入。
② 汇编指令代码line20-22,实现的功能是?
查阅资料得知0ah的是换行的ascii码。
该段代码的功能是打印一个换行
③ 汇编指令代码line24-30,实现的功能是?
打印输入的字符
实验任务3
assume cs:code, ds:data
data segment
x dw 91, 792, 8536, 65521, 2021
len equ $ - x
data ends
stack segment
dw 8 dup(?)
stack ends
code segment
start:
mov ax, data
mov ds, ax
mov ax, stack
mov ss, ax
mov sp, 16
mov si, offset x
mov cx, len/2;dw字数据,循环次数为len的一半
s: push cx
mov ax, [si]
mov dx, 0
mov cx, 0;计数器
add si, 2
;上述为入口参数
call printNumber
call printSpace
pop cx
loop s
mov ah, 4ch
int 21h
printNumber:
s1: mov bx, 10
div bx;每次提取一位
inc cx
push dx;每次将余数压入栈中
mov dx, 0
cmp ax, 0;判断商是否为零,为零则停止
jne s1
mov ah, 2
s2: pop dx;倒序输出,与字符的位数正好对应
or dx, 30h;转换成ASCII码
int 21h
loop s2
ret
printSpace:
mov ah, 2
mov dl, ' '
int 21h
ret
code ends
end start
运行测试截图:
这题用到的知识点有:cmp
指令的运用,跳转指令,以及32位除以16位的除法,具体的相关说明在代码中有相关注释。
实验任务4
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 cx, len
mov si, offset str
call strupr
mov ax, data
mov ds, ax
mov cx, len
mov si, offset str
s: mov ah, 2
mov dl, ds:[si]
int 21h
inc si
loop s
mov ah, 4ch
int 21h
strupr:
s0:
cmp byte ptr [si],'a'
jb s1
cmp byte ptr [si],'z'
ja s1
sub byte ptr [si],32
s1: inc si
loop s0
ret
code ends
end start
在debug中调试截图:
调用子程序strupr
之前:
调用之后:
结果:
这个题目对于cmp指令的要求更高,利用该指令,可以判断出字符串中那些属于小写字母,那些不属于小写字母,再把小写字母转换成大写字母,最后统一输出即可。
实验任务5
点击查看代码
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
测试截图:
输入7:
输入2:
分析:该程序的功能是判断输入的字符是否为7,是7则在显存中24行,70列处输出yes
,否则输出no
.
具体用到的子功能为:
int 21h的1号子功能:
mov ah, 1
int 21h ; (al) <-- 输入字符的ascⅡ码
int 21h的9号子功能:
mov ah, 9
mov ds, ×× ; ××是待输出字符串所在段的段地址
mov dx, ×× ; ××是待输出字符串第一个字符的偏移地址
int 21h
int 10h的2号子功能:
mov ah, 2
mov bh, ×× ; ××是页号
mov dh, ××
mov dl, ×× ; ××是列号
int 10h
实验任务6
调试结果:
编译成功,内中断程序可以使CPU在执行到特殊情况时调用,中断处理程序可以让CPU从错误中恢复。