汇编小知识
前言
这周做微机实验的时候,遇到一点小问题与疑惑?在此做一个小笔记,可能给朋友们一点小帮助。
代码环境是emu8086,不过代码不是关键,主要是对一些疑惑,自我整理出来的一些答案。
一点算法小知识——位运算
先欣赏一下这行汇编代码
AND AL,11011111B
功能小解释
这份代码完整的功能其实是将小写字母转大写输出,这段就是实现小写转大写的情况。前面以及实现了其他非小写字母字符的过滤,这是将小写转大写。
这里笔者在课上的时候迷惑了,这凭什么是这行代码,小写和大写字母在ASCII中相差32,那么直接减32即可,凭什么是这行代码。
查了一下资料知道 AND 其实就是位运算的意思,其实就是位与,就是c++ 中的 & ,按位与,然后我们再看源操作数 11011111B 其实就是32 = 2 ^ 5的位置刚好是0,做位与的结果其实就是相当于 -32。
而且作者一开始以为AND 运算不会影响 Flag 寄存器的值,其实还是会影响。
重头戏——寄存器保护
我们先写出以下代码,实现的功能是输入小写会转大写,输入T输出TEST,其他则直接输出字符。
DATA SEGMENT
MESSAGE DB 'THIS IS A SAMPLE PROGRAM OF KEYBOARD AND DISPLAY'
DB 0DH,0AH,'PLEASE STRIKE THE KEY!',0DH,0AH,'$'
DATA ENDS
STACK SEGMENT PARA STACK'STACK'
DB 50 DUP(0)
STACK ENDS
CODE SEGMENT
ASSUME CS:CODE,DS:DATA,SS:STACK
START: MOV AX, DATA
MOV DS,AX
MOV AX,STACK
MOV SS,AX
MOV DX,OFFSET MESSAGE
MOV AH,9
INT 21H
AGAIN: MOV AH,1
INT 21H
CMP AL,1BH
JE EXIT
cmp al,54H ;当累加寄存器是大写字母T的时候,跳转到自编程序EXE中
JE EXE
CMP AL,61H
JC ND
CMP AL,7AH
JA ND
AND AL,11011111B
ND: MOV DL,AL
MOV AH,2
INT 21H
JMP AGAIN
EXIT: MOV AH,4CH
INT 21H
EXE: MOV AH, 2 ;先将累加寄存器AX的高位AH为2的时候,表示输出
mov dl, 54h ;将数据寄存器存入大写字母T的值
INT 21H ;输出
mov dl, 45h ;存入大写字母E的值
int 21h
mov dl, 53h ;存入大写字母S的值
int 21h
mov dl, al ;存入T的值
int 21h
jmp again
CODE ENDS
END START
重点关注最后几行代码
mov dl, al
这里本来是 al 是直接用 54H代替的,但是作者突发奇想传个al寄存器应该也会这样。可是结果
run一下
寄!怎么会输出TESS啊?
不慌,出现问题,我们要冷静debug。
我们打开debug 工具,依旧传入 T,然后单步执行,在输出第二个字符的时候,AX的低位也就是AL变成了 45H 也就是E,这是为什么呢,此时不是输出嘛?为什么会把AL的值改变呢?
查阅过资料总结了一下,其实就是在输出的时候会将结果赋给AL,作为数据的保存。解决方式其实可以把这个寄存器PUSH到堆栈段实现一个对AX寄存器的保护。
后话
虽然实验报告可能写了一个晚上,有点久,但是还是很有收获的。