tsvico的博客

好的代码像粥一样,都是用时间熬出来的

汇编学习之路

写一些汇编基础,做个记录
包括输入输出、字符显示、字符判断等

输入输出先干为敬

1
2
3
4
5
MOV AH,1 ;输入字符
INT 21H ;输入后回显,键入字符的ascll送入AL
MOV DL,AL
MOV AH,2
INT 21H ;输出输入的字符

汇编

对于一个汇编程序有三段:数据段、堆栈段、代码段
初始化程序

1
2
mov AX,DATAS
mov DS,AX

为什么需要这个,因为mov无法实现将立即数传入段寄存器,但是mov可以将立即数传入通用寄存器
汇编语言程序中,数据的输入时以字符形式的0~9输入时对应的ASCII码减掉30H存入主存中的,输出时是加上30H
输出外设上的,比如十六进制A,主存中是0AH,加上37H变成了41H。因为主存中存的就是41H,所以可以直接输出带外设上

21H基本功能模块

1 从键盘上输入一个字符

AH=01H AL=输入自负的ASCII码值
举例

1
2
MOV AH,01H
INT 21H

2 向显示器输出一个字符

入口参数 AH=02H,DL=’输出字符串码’
举例

1
2
3
MOV DL,'C'
MOV AH,02H
INT 21H

3 向显示器 输出一个字符串

AH = 09H
DS = 欲输出字符串的段地址 DX = 欲输出字符串的 偏移地址

1
2
3
MOV  DX , OFFSET STRING  
MOV AH, 09H;
INT 21H

完整代码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21

DATAS SEGMENT
STRING DB 'HELLO WORLD!';数据段
DATAS ENDS

STACKS SEGMENT
DW 256 DUP(?)
STACKS ENDS
CODES SEGMENT
ASSUME CS:CODES,DS:DATAS
START:
MOV AX ,DATAS
MOV DS, AX ;这里开始初始化
;此处输入代码段代码
MOV DX, OFFSET STRING
MOV AH ,09H
INT 21H
MOV AH,4CH
INT 21H
CODES ENDS
END START

判断大小

CMP 比较.(两操作数作减法,仅修改标志位,不回送结果).
JA/JNBE 不小于或不等于时转移.
JAE/JNB 大于或等于转移.
JB/JNAE 小于转移.
JBE/JNA 小于或等于转移.
以上四条,测试无符号整数运算的结果(标志C和Z).
JG/JNLE 大于转移.
JGE/JNL 大于或等于转移.
JL/JNGE小于转移.
JLE/JNG小于或等于转移.
以上四条,测试带符号整数运算的结果(标志S,O和Z).

十六进制输出

示例

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
START:
MOV AX, DATA
MOV DS, AX

MOV BX, X
CALL LISTBX ;显示
MOV DL, 'H'
INT 21H

EXIT:
MOV AH, 4CH
INT 21H
;=============================
LISTBX:
MOV CX, 0404H
WR1:MOV AH, 2
ROL BX, CL
MOV DL, BL
AND DL, 15
CMP DL, 10
JB WR2
ADD DL, 7
WR2:ADD DL, '0'
INT 21H
DEC CH
JNZ WR1
RET

比较两个数字大小(完整代码)

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
DATA SEGMENT
DATA1 DB'比较无符号数据大小!',0DH,0AH,'$'
DATA3 DB'cx中的较大数为',0DH,0AH,'$'
;DATA4 DW'','$'

DATA ENDS
CODE SEGMENT
ASSUME CS:CODE,DS:DATA ; 说明代码段、数据段

Start:MOV AX,DATA
MOV DS,AX ;给DS赋值段
MOV DX,OFFSET DATA1
MOV AH,9
INT 21H ;显示提示”
mov DX,OFFSET DATA3
mov AH,9
INT 21H
Start1:
mov AX,12ABH
mov BX,4CBAH
CMP AX,BX
JAE Next
XCHG AX,BX
Next:
MOV CX,AX
Start2:
;下边是输出代码
;mov DATA4,CX
MOV BX,CX
CALL LISTBX ;显示
MOV DL,'H'
INT 21H

EXIT:
MOV AH, 4CH
INT 21H
;=============================
LISTBX:
MOV CX, 0404H
WR1: MOV AH, 2
ROL BX, CL
MOV DL, BL
AND DL, 15
CMP DL, 10
JB WR2
ADD DL, 7
WR2:
ADD DL, '0'
INT 21H
DEC CH
JNZ WR1
RET
CODE ENDS
END Start

压缩BCD码加减法

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
; 8086汇编
; 16位的压缩型BCD码的加减运算
; BUF = BCD1 + BCD2 - BCD3
; 加法 1234 + 7890 = 9124
; 减法 9124 - 5659 = 3465


DATA SEGMENT
BCD1 DW 1234H ;其中34是低8位
BCD2 DW 7890H ;其中90是低8位
BCD3 DW 5659H ;其中59是低8位
BUF DW 0
DATA ENDS
CODE SEGMENT
ASSUME CS:CODE,DS:DATA
START:
MOV AX,DATA
MOV DS,AX

CLC ;清除CF,因后面涉及进位标志CF
MOV AL,BYTE PTR BCD1
ADD AL,BYTE PTR BCD2 ;不带进位的加法
DAA ;DAA是压缩BCD加法的调整指令
MOV DL,AL
MOV AL,BYTE PTR BCD1+1
ADC AL,BYTE PTR BCD2+1 ;ADC是带进位的加法指令
DAA ;DAA是压缩BCD加法的调整指令
MOV DH,AL
SUB DL,BYTE PTR BCD3 ;不带进位的减法
MOV AL,DL
DAS ;DAS是压缩BCD减法的调整指令
MOV DL,AL
SBB DH,BYTE PTR BCD3+1 ;SBB是带借位的减法指令
MOV AL,DH
DAS ;DAS是压缩BCD减法的调整指令
MOV DH,AL
MOV BUF,DX ;将最后结果存入BUF

MOV AX,4C00H
INT 21H
CODE ENDS
END START

7/3号,写个简单的,同样是压缩BCD码加法

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
DATA SEGMENT
DATA ENDS
CODE SEGMENT
ASSUME CS:CODE,DS:DATA ; 说明代码段、数据段

Start:MOV AX,DATA
MOV DS,AX ;给DS赋值段
Start1:
MOV AX,6698H
MOV BX,2877H
ADD AL,BL
DAA
MOV CL,AL
MOV AL,AH
ADC AL,BH
DAA
MOV AH,AL
MOV AL,CL
MOV AH,4CH
INT 21H
CODE ENDS
END Start

记一下:数据寄存器AX、BX、CX、DX,每个数据寄存器都是16位,但是又可以把高、低8位分别作为两个独立的8位寄存器来用。高8位分别记作AH、BH、CH、DH,低8位分别记作AL、BL、CL、DL。例如AX可以当做两个8位寄存器AH、AL来用

8086/8088CPU的14个寄存器除了这四个16位可以分别当做两个8位,其他都不能

坚持原创技术分享,您的支持将鼓励我继续创作!