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
指令。
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,实现的功能是?
打印输入的字符
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位的除法,具体的相关说明在代码中有相关注释。
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指令的要求更高,利用该指令,可以判断出字符串中那些属于小写字母,那些不属于小写字母,再把小写字母转换成大写字母,最后统一输出即可。
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
调试结果:
编译成功,内中断程序可以使CPU在执行到特殊情况时调用,中断处理程序可以让CPU从错误中恢复。