资源描述
个人收集整理 勿做商业用途
第五章 程序设计方法和上机调试
§5.1 汇编语言程序设计方法
汇编语言设计基本要求:
①结构化、简明、易读、易调试、易维护(修改、扩充等)
②执行速度快:
③占用的存储空间少 相矛盾,根据实际权衡
汇编语言程序设计的基本步骤:
①分析问题,建数学模型,确定算法
②绘制流程图或写出步骤安排
③分配空间及寄存器,根据框图写程序
④静态检查
⑤上机调试
程序结构类型:
l 顺序结构
l 分支结构 不同的情况有不同的做法
l 循环结构 重复做的工作
l 子程序 多次重复使用的、具有通用性便于共享的部分。
子程序前面应加必要的说明
;子程序名称
;功能
;入口参数
;出口参数
;所用寄存器
例1:查平方表程序---- 顺序结构
0~15的平方值存在TABLE开始的16个单元.任给一自然数M在XX单元中。查表求M的平方值,并将结果存入YY单元中.
STACK SEGMENT PARA STACK ‘STACK'
DB 50 DUP(?)
STACK ENDS
; 组合类型,本段可以被别的段引用 类别
DATA SEGMENT PARA PUBLIC ‘DATA'
TABLE DB 0,1,4,9,16,25,36,49,64,81
DB 100,121,144,169,196,125
XX DB 6
YY DB ?
DATA ENDS
;
CODE SEGMENT
ASSUME CS:CODE,DS:DATA,SS:STACK
段寄存器指向说明CS指向CODE段……
START PROC FAR
PUSH DS
MOV AX,0 将DOS段值和偏移量(零)入栈。
PUSH AX
MOV AX,DATA
MOV DS,AX;当前数据段
MOV BX,OFFSET TABLE ;取表的偏移量
MOV AH,0 可用MOV AL,XX
MOV AL,XX XLAT 代替
ADD BX,AX ;取要查数的偏移量
MOV AL,[BX] ;查表MOV AL,XX [BX]
MOV YY,AL
RET
START ENDP
CODE ENDS
END START
P67页*:XLAT 换码指令 完成一字节的查表转换 AL←(BX+AL)
表的首址放入BX,相对首址偏移值入AL。
TABLE
[XX]
XX
YY
0
1
4
…
…
24
…
…
E1
6
24
例2: 数据块传送-——— 分支结构
将内存中首址为STG1、长度为STRSE的数据块传送到首址为STG2的区域中去。
无重迭 STG1+STRSE-1<STG2
或STG2+STRSE-1<STG1
有重迭时 STG1+STRSE-1≥STG2
STG2+STRSE-1≥STG1
分析可知 第一种情况用减量传送STG1+STRSE-1≥STG2,其它都用增量传送。
DATA SEGMENT
STRG DB 100 DUP(?)
STG1 EQU STRG+7
STG2 EQU STRG+25
STRSE EQU 50 ;传送50字节数据
DATA ENDS
;
STACK SEGMENT PARA STACK ‘STACK’
DB 100 DUP(?) ;定义100个字节堆栈
STACK ENDS
;
CODE SEGMENT
ASSUME CS:CODE,DS:DATA,ES:DATA,SS:STACK
GOO PROC FAR 主模块用START
PUSH DS
MOV AX,0 DOS入栈
PUSH AX
MOV AX,DATA 指向当前数据段
MOV DS,AX
MOV ES,AX
MOV CX,STRSE ;计数器值
MOV SI,OFFSET STG1;数据源指针
MOV DI,OFFSET STG2;目的数据指针
CLD ;增量传送
MOV BX,SI PUSH SI ;保护源操作数指针
ADD BX,STRSE-1 ADD SI,STRSE-1 ;源最后一个数指针
CMP BX,DI CMP SI,DI ;比较STG1+STRSE-1≥STG2.减
POP SI ;恢复SI <STG2。增(?)
JB OK ; 低于则转移
STD ;减量传送
ADD SI,STRSE-1 ;源尾部
ADD DI,STRSE-1 ;目的尾部
OK:REP MOVSB ;重复传送
RET
GOO ENDP
CODE ENDS
END GOO
例5—2 吴 分支结构
存储器中由一串字符串首址为BUF,字符串长度N小于256,要求分别计算出其中数字、字母和其它字符的个数,存放在字符串的下面三个单元中。
解:本题根据字符的ASCII码值范围进行判断,字母41H~5AH,数字30H~39H
DATA SEGMENT
BUF DB N
DB 01H,38H,47H,60H,…,76H
NUM DB 3 DUP(?)
DATA ENDS
;
STACK SEGMENT PARA STACK ‘STACK’
DB 100 DUP(?) ;定义100个字节堆栈
STACK ENDS
;
CODE SEGMENT
MAIN PROC FAR
ASSUME CS:CODE,DS:DATA, SS:STACK
START :PUSH DS
MOV AX,0 DOS入栈
PUSH AX
MOV AX,DATA 指向当前数据段
MOV DS,AX
MOV CH,N ;数组个数N CH
MOV BX,1
MOV DX,0 ;DH计数字个数,DL计字母个数
LP: MOV AH,BUF [BX] ;第一次取01H AH
CMP AH,30H
JL NEXT ;小于‘0’ 转
CMP AH,39H
JG ABC ;大于‘9'转
INC DH ;数字个数+1
JMP NEXT
ABC: CMP AH,41H
JL NEXT ;小于‘A’ 转
CMP AH,5AH
JG NEXT ;大于‘Z'转
INC DL ;字母个数+1
NEXT: INC BX ;数组地址+1
DEC CH ;计数—1
JNZ LP ;不为0跳转继续
MOV BUF [BX],DH ;数字个数送内存单元
MOV BUF [BX+1],DH ;字母个数送内存单元
MOV AH,N
SUB AH,DH ;计算其它字符个数
SUB AH,DL
MOV BUF [BX+2],AH ;其它字符个数送内存单元
RET
MAIN ENDP
CODE ENDS
END START
P124 例6 在一个带符号的字类型的数据串中,找出最大值---循环结构
STT SEGMENT PARA STACK ‘STACK‘
DB 50 DUP(?)
STT ENDS
;
DATA SEGMENT
VALUES DW 2,21,87,-23,14,90;0002H,0015H,0057H,FFE9H,000EH,005AH
COUNT EQU $-VALUES ;数据串长度字节数,$-(地址计数器值)
YY DW ?
DATA ENDS
;
CODE SEGMENT
ASSUME CS:CODE,DS:DATA,SS:STT
START PROC FAR
PUSH DS
MOV AX,0 ;DOS交权时,DOS的段值和偏移量入栈.
PUSH AX
MOV AX,DATA 指向DS段(当前)
MOV DS,AX
MOV BX,OFFSET VALUES ;数据首址偏移量
MOV CX,COUNT/2-1 ;量计数器,比较n-1次
MOV AX,[BX] ;取第一个放AX中,做初始最大值。
FINDM:ADD BX,2 ; 下移一个数据
CMP AX,[BX] ;原来的最大值与现在的比
JG TESTIP ;若原来的大、不变、转移
MOV AX,[BX] ;否则,换新的最大值入AX
TESTIP: LOOP FINDM ;检查 循环是否结束
MOV YY,AX ; 将最大值放入单元
RET
START ENDP
CODE ENDS
END START
方法:将第一个数据作为初始最大值,再将后面的数据与其比较,将最大值放入AX中。
P131页 例10.在数据段定义了两个数组,编一个程序实现对两数组分别求和。要求采用子程序调用形式。
调用程序用CALL指令调用之前把子程序所需参数入栈,以便子程序运行时使用这些参数.当子程序返回后,这些参数已不再有用,因此子程序结束时的RET指令应使用带常数的返回指令。以便堆栈恢复原态.
STACK SEGMENT PARA STACK ‘STACK‘
SPAE DW 20 DUP(?)
TOP EQU LENGTH SPAE
STACK ENDS
;
DATA SEGMENT
ARY1 DB 1,2,3,4,5,6,7,8,9,10
NUM1 EQU $-ARY1 ;求数组1长度
SUM1 DW ?
ARY2 DB 20,30,40,10,21,33
NUM2 EQU $-ARY2 ;求数组2长度
SUM2 DW ?
DATA ENDS
;
MAIN SEGMENT
ASSUME CS:MAIN,DS:DATA,SS:STACK
STR:PUSH DS
MOV AX,0 DOS入栈
PUSH AX
MOV AX,DATA 指向当前数据段
MOV DS,AX
MOV AX,NUM1
PUSH AX ;数组长度进栈。参数1
MOV AX,OFFSET ARY1
PUSH AX ;数组首址偏移量进栈,参数2
CALL FAR PTR SUM ;CS、IP先后进栈
MOV AX,NUM2
PUSH AX
MOV AX,OFFSET ARY2
PUSH AX
CALL FAR PTR SUM
HLT
MAIN ENDS ;主程序结束
;功能:数组求和;入口参数:数组长度、首址偏移量。出口:∑
;所用寄存器AX、BX、CX、BP
PROCE SEGMENT
ASSUME CS:PROCE ,DS:DATA,SS:STACK
SUM PROC FAR
PUSH AX
PUSH BX
PUSH CX
PUSH BP
MOV BP,SP
PUSHF
MOV CX,[BP+14] ;参数1,因在PUSHF之前SP移入BP,所以+14是参1。
MOV BX,[BP+12] ;参数2,首址偏移量
MOV AX,0
ADN: ADD AL,[BX]
INC BX
ADC AH,0 ;进位加进去
LOOP ADN
MOV [BX],AX ;数据之和送结果单元
POPF ;标志出栈 Flag←(sp+1,sp),sp←sp+2
POP BP
POP CX
POP BX
POP AX
RET 4 ;舍弃两个参数返回原来参数入栈两次。
SUM ENDP
PROCE ENDS
END STR
§5。2 汇编语言上机调试过程
⑴进入用户目录进行程序编辑.
>login
F:\>cd user
F:\〉user\5
F:\〉user\5\Edit
保存: Alt进入主菜单.Save。 save as…
xyz.ASM.
⑵MASM
Source filename[。ASM]:文件名 源文件
Object filename[xyz.OBJ]: 目标文件
Source Listing [NUL。LST]: 打印文件
Cross Reference[NOL.CRF]: 交叉索引文件
0 Warning Errors
0 Severe Errors
DOS状态下 TYPE XYZ.ASM
LST 均可打印出来
⑶连接 Link
汇编程序建立的目标码文件即.obj文件必须经过连接以后,才能成为可执行文件.即。EXE文件。
Object Modules[.OBJ]:xyz 目标文件
RON File[xyz.EXE]: 可执行文件
List File[NUL。MAP]: 内存分配图文件
Libraries [。LIB]: 库文件
若有多个要连接的OBJ文件,应一次打入,各OBJ文件名词用“+”号相间隔。
⑷运行文件 XYZ
⑸调用DEBUG程序调试
在\DOS\DEBUG[Path][filemame].
\user\5\xyz.exe
即出现“一"提示符,说明系统已在DEBUG管理之下。
一、 显示内存单元命令。
D 或D[地址] 从地址在显示80H个字节。
当前地址开始 或—D[范围]
显示80H字节
二、修改存储单元内容命令E
E 地址 显示原来单元内容
输入一个字节的16进制数,按空格键,则修改完成。接着显示下一单元的地址和原有内容。按回车结束此命令。
三、检查和修改寄存器内容的命令R
—R 显示所有寄存器内容
—R AX 显示AX寄存器内容
-RF 显示和修改标志位状态
置位:OY DN EI NG ZR AC PE CY-
复位:NV UP DI PL NZ NA PO NC
除跟踪标志T之外。
四、 运行命令 G
-G [=执行的起始地址偏移量][断点地址1[断点地址2]]
最多可设置10个断点。
五、跟踪命令T(单步)
—T[=指定地址] 执行指定地址处的一条指令停下来,显示CPU所有寄存器和标志状态。
—T 从CS:IP执行一条指令
—[=指定地址][值]
执行的指令条数。
六、 反汇编命令U
-U 显示32个字节的源程序内容
—U 范围
P118页例:
DATA SEGMENT
TABLE DB 0,1,4,9
XX DB 2
YYO DB ‘RESULT:’
DATA ENDS
;
CODE SEGMENT
ASSUME CS:CODE,DS:DATA,SS:STACK
段寄存器指向说明CS指向CODE段……
START PROC FAR
PUSH DS
MOV AX,0 将DOS段值和偏移量(零)入栈。
PUSH AX
MOV AX,DATA
MOV DS,AX;当前数据段
MOV BX,OFFSET TABLE ;取表的偏移量
MOV AH,0 ;可用MOV AL,XX
MOV AL,XX ; XLAT 代替
ADD BX,AX ;取要查数的偏移量
MOV AL,[BX] ;查表MOV AL,XX [BX]
MOV YY,AL
RET
START ENDP
CODE ENDS
END START
…
MOV AL,[BX]
ADD AL,30H ;转换成ASCII码值
MOV YY,AL
加 MOV AH,9
MOV DX,OFFSET YYO
INT 21H
RET
§5。3 宏汇编与条件汇编
源程序中若有的程序段要多次使用,可不重复书写,而用一条宏指令来代替。
格式:宏指令名 MACRO (形式参量表)
…
宏定义体
ENDM
宏调用格式:宏指令名 实元表.
例:8位带符号数相乘,结果放入RESULT单元
被乘数 乘数 结果
宏定义 MULTPLY MACRO OPRD1 ,OPRD2, RESULT
PUSH AX
MOV AL,OPRD1
IMUL OPRD2 ;AX←AL*OPRD2
MOV RESULT,AX
POP AX
ENDM
宏调用:MULTPLY 240,BL,SAVC
…
宏展开:即在汇编时产生
+ PUSH AX
+MOV AL,240
+IMUL BL
+MOV SAVC,AX
+POP AX
…
实元可以是常数、寄存器、存储单元名及地址或表达式也可以是操作码。
例 移位指令都以S打头,后面可以用宏。
除了MACRO之外,还有宏汇编伪指令REPT、IRP、IRPC也都以ENDM作为结束符.
二. PURGE(用来取消宏定义)与LOCAL
格式:PURGE 宏指令名[,…]
LOCAL 用来说明宏体内的标号为局部标号,以免多次调用宏定义时标号重复而出错。
格式: LOCAL 局部标号表
三。 宏指令与子程序的区别
①宏指令是为了简化源程序的书写,汇编时,把宏定义体插入到宏调用处。所以宏指令并没有简化目标程序,不节省目标程序所占的内存单元。
②子程序方法简化了目标代码,节省空间,但每次调用都要保护现场而后恢复,这样执行时间长点,速度稍慢。
③选用时根据实际情况,速度是主要矛盾时,用宏指令;存储空间是主要矛盾时选用子程序方法。
四。 条件汇编
根据条件决定一段源程序是否在该程序中起作用,起作用的则将其生成目标程序代码,否则,不生成。
多种格式:
IFXX 条件表达式
… (语句体1)
[ELSE]
(语句体2)
…
ENDIF
有:IF、IFE、IFDEF、IFNDEF、IFB、…IFNB、…
计算条件:为真时汇编。
展开阅读全文