Bootstrap

《汇编语言程序设计教程》人民邮电出版社第二版习题及参考答案

网上的答案是第一版的,重新整理了一下

《汇编语言程序设计教程》人民邮电出版社第二版
刘慧婷,王庆生 主编
习题及参考答案

更多汇编内容请访问:omegaxyz.com

第一章至第五章

核对及编辑:xyjigsaw

习题1
1.1 什么是机器语言?什么是汇编语言?简述汇编语言的特点。
机器语言就是用二进制编码组成的机器指令的集合和一组使用机器指令的规则。汇编语言是对机器指令中的操作码用英文单词的缩写描述,对操作数用标号、变量、常量描述。
汇编语言的特点:
(1)与机器有关:移植性差,但可直接控制硬件。
(2)程序效率高。
(3)局限性:受指令的限制,如考虑存储单元等。
(4)调试困难。

1.2 汇编程序与汇编源程序的区别是什么?
汇编源程序是指用汇编语言编写的程序,而汇编程序特指将汇编源程序汇编成目标文件的编译程序。
1.3 把下列十进制数转换为二进制数和十六进制数。
(1)67 (2)34 (3)254 (4)123
答:(1)1000011,43 (2)100010, 22
(3)1111 1110,FE (4)111 1011,7B

1.4 把下列二进制数转换为十六进制数和十进制数。
(1)01101101 (2)10110010 (3)111111
答:(1)6D,109 (2)B2,178, (3)3F,63

1.5 作下列十六进制数的运算,并转换为十进制数校核。
(1)5A+64 (2)86-49 (3)123-9A (4)43×2B
答:(1) BE (2)3D (3) 89 (4) B41

1.6 根据补码定义把下列十进制数表示为8位二进制补码。
(1) 64 (2) -24
答:(1)0100 0000 (2)[10011000]原=[1110 1000]补

1.7 下列各数均为十进制数,请采用8位二进制补码运算,并回答标志寄存器FLAGS中CF和OF的值,运算结果所代表的十进制数是多少?如果用16位二进制补码运算,其结果所代表的十进制数是多少?FLAGS中CF和OF的值呢?
(1)85+69 (2)85+(-69) (3)85-(-69) (4)85-(69)
答:85=55H,69=45H,-69=BBH,
8位二进制补码运算:
(1)85+69 =55H+45H=9AH=154, CF=0,OF=1
(2)85+(-69)=55H+BBH=10H=16,CF=1,OF=0
(3)85-(-69)=55H-BBH=9AH=154,CF=1,OF=1
(4)85-(69)=55H-45H=10H=16,CF=0,OF=0

16位二进制补码运算:
85=0055H,69=0045H,-69=0FFBBH,
(1)85+69 =0055H+0045H=009AH=154, CF=0,OF=0
(2)85+(-69)=0055H+0FFBBH=0010H=16,CF=1,OF=0
(3)85-(-69)=0055H-0FFBBH=009AH=154,CF=1,OF=0
(4)85-(69)=0055H-0045H=0010H=16,CF=0,OF=0

2.1 简述计算机系统组成。
答:计算机由中央处理器CPU,存储器,输入系统,输出系统组成,由系统总线连接在一起。CPU包括运算器和控制器,运算器执行指令,控制器负责计算机的控制。存储器是计算机的记忆部件,以二进制形式存放程序和数据。输入输出系统包括大容量存储器,如硬盘,以及其他外设,如鼠标,键盘,显示器等。

2.2简述16位机的各类寄存器的主要作用。
答:(1)数据寄存器:Ax,Bx,Cx,Dx; AX:作为累加器,是算术运算的主要寄存器。在乘除等指令中存放操作数,在I/O指令中使用它与外部设备传送信息。BX:当通用寄存器使用,在计算存储器地址时,作基址寄存器使用。CX:当通用寄存器使用,此外常用来保存计数值,当计数器使用。
DX:当通用寄存器使用,一般在作双字长运算时把DX和AX组合使用,对某些I/O操作,DX用来存放I/O的端口地址。
(2)地址寄存器:Sp,Bp,Si,Di
(3)段寄存器:Cs,Ds,Es,Ss;段寄存器的作用是专用于存储器寻址,用来直接或间接地存放段地址。
(4)专用寄存器:Ip,Flags;Ip寄存器专门存放下一条指令的地址,Flags标志寄存器,又称程序状态寄存器。它是存放条件码标志、控制标志和系统标志的寄存器。

2.4 实模式下,写出段地址和偏移地址为1234:2002、1430:0042、FF00:0FFF的物理地址。
答:
1234:2002=12340+2002=14342
1430:0042=14300+0042=14342
FF00:0FFF=FF000+0FFF=FFFFF
2.8
52506=50000+2506,=52500+0006,基地址最大为5250,最小为5000.
段地址取值范围:0000—FFFF。既65536个。
2.9 从物理地址为00100H开始到00103H单元中顺序存放的数据为:12H,34H,56H,78H。请画出数据存放示意图,并回答以下问题:
(1) 写出00101H字节单元的内容
(2) 写出00102H字单元的内容
答案:
(1)(00101)= 34H
(2)(00102)= 7856H
习题3
3.1 写出从汇编语言源程序的建立到产生可执行文件的步骤和上机操作命令。
答:(1)用编辑程序EDIT建立 .ASM源文件
(2)用汇编程序MASM把.ASM文件原文件汇编成.OBJ文件
(3)用连接程序LINK将.OBJ文件转换成.EXE文件
(4)在DOS下直接运行.EXE文件或在DEBUG下调试该.EXE文件

3.2 列表文件.LST是在什么阶段产生的?列表文件.LST中有哪些内容?
答:.LST 在汇编得到.obj的同时得到,列表文件报告了汇编过程中产生的很多有价值的参考信息。主要包括源程序和机器语言清单、指令和变量的偏移地址等等。

3.3 写出定义一个代码段的语句,段名为MYPRG。
答:MYPRG segment
Assume cs:MYPRG
start:
MYPRG ends
End start

3.4 程序中用什么语句来结束程序的执行?用什么语句来表示程序的结束和指出程序执行的起点?
答:程序的结束:mov ah,4ch
int 21h
程序的结束和指出执行的起点:End start

3.5 汇编语言源程序的文件扩展名是什么?把它改为.EXE扩展名后,可以认为是可执行程序吗?
答:源程序的文件扩展名为*.asm,改扩展名不可以执行。

3.6 列出子目录C:\YOUPRG下的扩展名为.ASM的所有文件,在D盘根目录下建立一个子目录MYPRG,并进入子目录MYPRG,再把C:\YOUPRG下的文件YOU.ASM复制到D:\MYPRG下。写出完成以上要求的DOS命令。
答:(1)列出子目录:C:\YOUPRG>dir *.asm
(2)建立子目录:D:>MD MYPRG
(3)复制: C:\YOUPRG>copy YOU.asm D:\MYPRG

3.7 下图为DEBUG调入的可执行程序,回答以下问题:
(1) 程序的起始物理地址是多少?结束地址是多少?
(2) CS寄存器的值是什么?
(3) 程序的功能是什么?
(4) 写出查看DS:0处内容的DEBUG命令。
(5) 程序中2处的INT 21指令执行后有什么结果?
(6) 如果要运行这个程序应该用什么DEBUG命令?
(7) DEBUG命令-T=0 4 之后,寄存器AX、DS、DX的值是多少?

答:(1)0B63:0000
(2)0B63
(3)显示0B62:0000存储的字符串
(4)-d 0B62:0000
(5)第一处显示字符串,第二处结束程序
(6)-g
(7)(AX)=0962,(DS)=0B62,(DX)=0000

3.8 解释DEBUG程序中的如下调试命令。
– D , – E, – T, – G, – A , – R
答:- D 显示内存内容
– E 修改内存单元内容
– T 跟踪命令
– G 运行命令
– A 汇编命令
– R 查看或修改寄存器内容

3.9 用DEBUG调入PROG.EXE后,若程序列出如下:
1234:0100 MOV BX, [4000] 1234:0104 MOV AX, [BP] 1234:0106 MOV AH, 1
1234:0108 INT 21
1234:010A MOV DL, AL
1234:010C MOV AH, 2
1234:010E INT 21
1234:0110 RET
列出上面程序的DEBUG命令是( )。
寄存器CS的值为( ),第一条指令的物理地址为( )。
如果要修改寄存器BX为1200H,应键入DEBUG命令( )。
若要修改第二条指令中的BP为BX,应键入DEBUG命令( ) 。
答:(1) U
(2) 1234H
(3) 12440H
(4) R BX
(5) A [0104]

3.10 简述DOS系统功能INT 21H调用方法。
答:(1)将调用功能的功能号存入AH寄存器。
(2)如必要,设置该调用功能的入口参数。
(3)执行INT 21H指令。
(4)如必要,按规定取得出口参数(返回参数)。

3.11 Debug命令调试含有“INT 21H”命令的程序段时,如何实现单步执行,“-t”命令为何无法实现?
答:“INT 21H”指令是Dos的系统调用,如果使用“-t”跟踪,则进入DOS系统子程序。如果希望单步执行,可以通过P命令或“-g=x x+1”来实现。

3.12 有主存数据段存放情况如下图所示,请写出代码,输出数据段的字符串“inspire a generation!”。

答: mov ax,145B
mov ds,ax
mov ah,09
mov dx,0
int 21h

习题4
4.1 何为段地址?何为有效地址? 何为物理地址?
答:

例如DS=6542H,指令mov ax,DS:[123A]; 123A为有效地址,6542H为段地址,65420H+0123AH=6665A即物理地址.

4.2 指出以下指令的寻址方式,array是变量。
(1)mov ax, 9
(2)mov byte ptr[bx],9
(3)mov bx,[di] (4)mov ax,bx
(5)mov [si+bx],9
(6)mov array[bx],cx
(7)mov ax, array+9
(8)mov ax, array[bx+di] 答: (1)立即数寻址
(2)寄存器间接寻址
(3)变址寻址
(4)寄存器寻址
(5)基址变址寻址
(6)相对基址寻址
(7)直接寻址
(8)相对基址变址寻址

4.3 假定(DS)=1200H,(SS)=4400H, (BX)=463DH,(BP)=2006H, (SI)=6A00H,位移量D=4524H,以AX寄存器为目的操作数,试写出以下各种寻址方式下的传送指令,并确定源操作数的有效地址EA和物理地址。
(1)立即寻址
(2)直接寻址
(3)使用BX的寄存器寻址;无EA
(4)使用BX的间接寻址
(5)使用BP的寄存器相对寻址
(6)基址变址寻址
(7)相对基址变址寻址
答:(1)操作数在指令中无EA
(2)直接寻址:EA=4524H,物理地址=DS:4524
(3)使用BX的寄存器寻址:无EA
(4)使用BX的间接寻址:EA=463DH,物理地址=DS:463D
(5)使用BP的寄存器相对寻址:MOV AX,[BP+4524],EA=2006+4524,
物理地址=SS:EA
(6)基址变址寻址: MOV AX,[BX+SI],EA= BX+SI,物理地址=DS:EA
(7)相对基址变址寻址:MOV AX,[4524+BX+SI],EA=4524+BX+SI,
物理地址=DS:EA

4.4 在数据段定义了ARRAY数组,其中依次存储了五个字数据,ARRAY的起始地址(第一个数据的地址)为24H,请用不同寻址方式的指令,把第5个字送AX寄存器,指令条数不限。
答:(1)直接寻址
MOV AX,ARRAY+8 或MOV AX,[ARRAY+8] (2)使用BX的间接寻址
LEA BX,ARRAY+8 ;MOV AX,[BX] (3)使用BX的寄存器相对寻址
LEA BX,ARRAY ;MOV AX,[BX+8] (4)基址变址寻址
LEA BX,ARRAY ; MOV SI,8 ; MOV AX,[BX+SI]

习题5
5.1 溢出标志OF与进位标志CF有何作用和区别?
答:处理器对两个操作数进行运算时,按照无符号数求得结果,并相应设置进位标志CF;同时,根据是否超出有符号数的范围设置溢出标志OF。应该利用哪个标志,则由程序员来决定。也就是说,如果将参加运算的操作数认为是无符号数,就应该关心进位;认为是有符号数,则要注意是否溢出。

5.2 带符号数比较大小,当AX < BX时程序转向标号L1,若前导指令为CMP AX, BX,后续指令应为什么?若视为二个无符号数比较大小, 后续指令应为什么?
答:带符号数:JL L1, 无符号数:JB L1

5.3 说出CALL指令和INT 21H指令的操作,有何异同?
答:CALL把断点压入堆栈,不一定是远调,INT 21H还要把FLAGS压入堆栈,且是远调,总入口地址为84H内存中的两个字。

5.4 除了用4CH号系统功能调用结束程序的执行并退出,还有哪些办法?
答:RET指令,INT 20H,但要正确使用。

5.5 V是变量,指出下列错误的指令,说出错误原因,并给出合适的修改。
(1) MOV AX,[DX] (2) MOV DS,DATA
(3) MOV CS,AX
(4) MOV AX,DL
(5) PUSH AL
(6) ADD [BX],[DI] (7) LEA [BX],V
(8) MOV [DX],OFFSET V
(9) MOV [SI],2
(10) MUL BX,CX
(11) DIV 5
(12) MOV BYTE[SI],AX
(13) MOV AX,[SI+DI] (14) SHR AX,4
(15) CMP 6,AX
(16) MOV [FFFF],AX
(17) MOV AX,BX+4
(18) JMP FAR PRO
答: (1) MOV AX,[BX] (2) MOV DS,BX
(3) MOV ES,AX
(4) MOV AL,DL
(5) PUSH AX
(6) ADD [BX],DI
(7) LEA BX,V
(8) MOV DX,OFFSET V
(9) MOV WORD PTR [SI],AX
(10) MUL BX
(11) DIV BX
(12) MOV BYTE PTR [SI],2
(13) MOV AX,[BX+SI] (14) SHR AX,1
(15) CMP AX,6
(16) MOV [FFFE],AX
(17) MOV AX,[BX+4] (18) JMP FAR PTR PRO

5.6 在数据段定义了ARRAY数组,其中依次存储了4个字数据,根据以下要求把第4个字送AX寄存器。
(1)直接寻址
(2)使用BX的间接寻址
(3)使用BX和 ARRAY的寄存器相对寻址
(4)基址变址寻址
(5)MOV以外的其它指令
答:(1)直接寻址
MOV AX,ARRAY+6 或MOV AX,[ARRAY+6] (2)使用BX的间接寻址
LEA BX,ARRAY+6 ;MOV AX,[BX] (3)使用BX和 ARRAY的寄存器相对寻址
LEA BX,ARRAY ;MOV AX,[BX+6] (4)基址变址寻址
LEA BX,ARRAY ; MOV SI,6 ; MOV AX,[BX+SI] (5)MOV以外的其它指令
SUB AX,AX; ADD AX,[ARRAY+6]

5.7 画出数据在数据段中的存放情况,程序执行后,BX、DI、CX、DX寄存器中的内容是什么? 程序如下:
data segment
array dw 20,30,40,20h,30h,-6
buff db ‘abcd dataendscodesegmentassumecs:code,ds:datastart:movax,datamovds,axmovbx,array+1movdi,offsetarraymovcx,[di+5]movdl,buff+3movah,4chint21hcodeendsendstart14001E00280020003000FAFF61626364
(BX)=1E00H,(DI)=0000H,(CX)=2000H,(DX)=0064H

5.8 在DEBUG下设置(SP)=20H,设置AX、BX、CX、DX为不同值,把这四个寄存器内容依次压入堆栈,再从堆栈中依次弹出到SI、DI、BP、BX寄存器。写出一段程序实现上述操作,并画出每条入栈指令执行后SP和堆栈中数据的变化。
答:
code segment
assume cs:code
start:
push AX
push BX
push CX
Push DX
Pop SI
Pop DI
Pop BP
Pop BX
Hlt
Code ends
End start

5.9 求出7450H与以下各十六进制数的和及差,并根据结果标出SF、ZF、CF、OF标志位的值。
(1) 1234H (2)5678H (3)9804H (4)E0A0H
答: SF、ZF、CF、OF
(1)1234H+7450H, 1 0 0 1
(2)5678H+7450H, 1 0 0 1
(3)9804H+7450H, 0 0 1 0
(4)E0A0H+7450H, 0 0 1 0

5.10 在数据段有32位的无符号数变量X,Y,按如下格式定义,其中‘?’请用数值代替,用16位指令按要求写出程序。
X DW ?,?
Y DW ?,?
Z DW ?,?,?,?
(1)Z=X+Y。
(2)Z=X-Y。
(3)Z=|X-Y|。
(4)Z=X×Y。
答:(1)X和Y两个字数据相加,和存放在Z中。
MOV AX,X
MOV DX,X+2
ADD AX,Y
ADC DX,Y+2
MOV Z,AX
MOV Z+2,DX
(2)MOV AX,X
MOV DX,X+2
SUB AX,Y
SBB DX,Y+2
MOV Z,AX
MOV Z+2,DX
(3)MOV AX,X
MOV DX,X+2
SUB AX,Y
SBB DX,Y+2
TEST AX,8000H
JZ Exit
NOT AX
NOT DX
ADD DX,1
ADC AX,0
Exit: MOV Z,AX
MOV Z+2,DX
(4)MOV AX,X
MUL Y
MOV Z,AX
MOV Z+2,DX
MOV AX,X+2
MUL Y
ADD Z+2,AX
ADC DX,0
MOV Z+4,DX
MOV AX,X
MUL Y+2
ADD Z+4,AX
ADC DX,0
MOV Z+6,DX
MOV AX,X+2
MUL Y+2
ADD Z+6,AX
ADC DX,0
MOV Z+8,DX

5.11用移位指令为主实现对AX中的无符号数乘以5,不考虑乘积可能超出16位。
答: MOV DX,AX
MOV CL,2
SHL AX,CL
ADD AX,DX

5.12 用移位指令为主实现对AX中的无符号数乘以5,考虑乘积可能超出16位的情况。
答: MOV DX,0
MOV BX,AX
SHL AX,1
ADC DX,0
SHL DX,1
SHL AX,1
ADC DX,0
ADD AX,BX
ADC DX,0
5.13 把AX中的内容依次倒排序,即第0位移到第15位,第1位移到第14位,…。
答:先设AX值
Mov di,0
Mov bx,0
Mov si,1
Mov cl,2
Ror ax,1
K: Rol ax,cl
Mov bx,ax
And bx,si
Add di,bx
Shl,si,1
Cmp si,0
Jnz k
Mov ax,di

5.14 在数据段有如下定义:
BUFF DB ‘ABCD EFGHIJK
STR1 DB 12 DUP(?)
LEN DB ?
用串指令编写程序完成以下操作:
(1) 对字符串STR1全部置‘*’符。
(2) 从左到右把BUFF中的字符串传送到STR1。
(3) 从右到左把BUFF中的字符串传送到STR1。
(4) 比较BUFF与STR1两个字符串是否相等,如相等则DX=1,否则DX=0。
(5) 查找BUFF中有无字符 出现的次数计入BX寄存器。
答:
(1)
MOV AX,DATA
MOV DS,AX
MOV AL,‘*’
LEA DI,STR1
MOV CX,STR1-BUFF
CLD
REP STOSB
(2)
MOV AX,DATA
MOV DS,AX
MOV ES,AX
CLD
LEA SI,BUFF
LEA DI,STR1
MOV CX,STR1-BUFF
REP MOVSB
(3)
MOV AX,DATA
MOV DS,AX
MOV ES,AX
STD
LEA SI,STR1-1
LEA DI,LEN-1
MOV CX,STR1-BUFF
REP MOVSB

(4)
MOV AX,DATA
MOV DS,AX
MOV ES,AX
CLD
LEA SI,BUFF
LEA DI,STR1
MOV CX,STR1-BUFF
REPE CMPSB

(5)
MOV AX,DATA
MOV ES,AX
MOV BX,0
CLD
MOV AL,‘$’
LEA SI,BUFF
MOV CX,STR1-BUFF
NEXT: REPNE SCASB
JCXZ NO-FOUND
INC BX
JMP NEXT

5.15 对于给定的AX和BX的值,执行下列程序段,程序将转向哪里?
ADD AX, BX
JNO L1
JNC L2
SUB AX,BX
JNC L3
JNO L4
JMP L5
(1) AX=1234H, BX=6789H
(2) AX=790EH, BX=8831H
(3) AX=E002H, BX=8086H
答:L1,L1,L1,L5

5.16 下面不完整的程序段是比较AX和BX的值,把其中大的数送MAX变量。如果是无符号数,应如何填写指令?如果是有符号数,应如何填写指令?
CMP AX,BX
( )
MOV MAX,AX
( )
L1:MOV MAX,BX
L2:HLT
答: JB L1
JMP L2

5.17 在下列程序段的括号中分别填入如下指令,程序执行完后,AX、CX的内容是什么?
(1) LOOP L1
(2) LOOPE L1
(3) LOOPNZ L1
MOV AX, 6
MOV CX,3
L1:ROL AX,CL
TEST AL,3
( )

答:(1)AX=0180H,CX=0000
(2)AX=0003H,CX=0002
(3)AX=0180H,CX=0000

5.18 测试AL寄存器,如果最高位和最低位同时为0,则转L0,如果最高位和最低位只有一位为1,则转L1,如果最高位和最低位同时为1,则转L2。画出流程图,并编写程序段。
答:
Code segment
Assume cs:code
Start:

ROR AL,1
AND AL,3
JZ L0
SUB AL,3
JZ L2
JMP L1
L0:
JMP Exit
L1:
JMP Exit
L2:
Exit:hlt
code ends
end start

5.19 从键盘输入一个英文字母,显示输出其大写。画出流程图,并编写程序段。
答:
Code segment
Assume cs:code
Start:
Mov ah,1
Int 21h
sub al,20h
mov dl,al
mov ah,2
int 21h
hlt
code ends
end start

5.20 从键盘输入一位数字N,显示输出N个N。画出流程图,并编写程序段。
答:
CODE SEGMENT
ASSUME CS:CODE
START:
MOV AH,1
INT 21H
MOV DL,AL
AND AL, 0FH
MOV CL,AL
MOV CH,0
NEXT: MOV AH,2
INT 21H
LOOP NEXT
MOV 4CH
INT 21H
CODE ENDS
END START

5.21 在数据段有压缩的BCD码表示的十进制数,写出指令分别完成十进制加法A+B,K+J和减法A-B,K-J。结果放在AX,回答AX的内容。
A DB 65H,
B DB 37H
K DB 98H
J DB 69H
答:(1) MOV Al,A
MOV Bl,B
ADD AL,BL
AAA
SUB AL,BL
DAS
(2) MOV Al,K
MOV Bl,J
ADD AL,BL
AAA
SUB AL,BL
DAS
(1)A+B=9CH,AX=0102H.A-B=2EH,AX=0028H
(2)K+J=01H,AX=0107H.K-J=2FH,AX=0029H

5.22 用相对基址变址寻址方法求ARRAY行列式的值。
ARRAY DW 2,3,5
DW 1,4,6
DW 2,7,9
答:
data segment
Array dw 2,3,5
dw 1,4,6
dw 2,7,9
data ends
code segment
assume cs:code,ds:data
start:
mov ax,data
mov ds,ax
mov ax,0
mov bx,0fffah
a: mov si,0
add bx,6
cmp bx,13
jg d

b: add ax,array[bx][si] cmp si,3
jg a
add si,2
jmp b
d: hlt
code ends
end start

习题6

6.1 画图说明下列数据定义语句所示内存空间的数据,并回答寄存器的值。

ORG 0

ARRAY LABEL BYTE

DA1 DW 2,9,14,3,315H,-6

DA2 DB 7,‘ABCDEDFG’

LEN = $-DA2

ORG 100H

DA3 DW DA4

DA4 DB 4 DUP(2 DUP(1,2,3),4)

。。。。。。

MOV AL,ARRAY+2 (AL)=( )H

ADD AL,DA2+1 (AL)=( )H

MOV AX,DA2-DA1 (AX)=( )H

MOV BL,LEN (BL)=( )H

MOV AX, DA3 (AX)=( )H

MOV BX, TYPE DA4 (BX)=( )H

MOV BX, OFFSET DA4 (BX)=( )H

MOV CX, SIZE DA4 (CX)=( )H

MOV DX, LENGTH DA4 (DX)=( )H

MOV BX, WORD PTR DA4 (BX)=( )H

MOV BL, LEN AND 0FH (BL)=( )H

MOV BL, LEN GT 5 (BL)=( )H

MOV AX, LEN MOD 5 (AX)=( )H

答:

MOV AL,ARRAY+2 (AL)=( 09 )H

ADD AL,DA2+1 (AL)=( 41 )H

MOV AX,DA2-DA1 (AX)=( 000c )H

MOV BL,LEN (BL)=( 09 )H

MOV AX, DA3 (AX)=( 0102 )H

MOV BX, TYPE DA4 (BX)=( 0001 )H

MOV BX, OFFSET DA4 (BX)=( 0102 )H

MOV CX, SIZE DA4 (CX)=( 0004 )H

MOV DX, LENGTH DA4 (DX)=( 0004 )H

MOV BX, WORD PTR DA4 (BX)=( 0201 )H

MOV BL, LEN AND 0FH (BL)=( 09 )H

MOV BL, LEN GT 5 (BL)=( ff )H

MOV AX, LEN MOD 5 (AX)=( 0004 )H

6.2 变量和标号有哪些区别?变量和标号有哪些属性?如何获取属性值?写出指令。

答:变量是为指令提供的操作数,标号是为指令提供标识,都是为了在指令中引用。它们最主要的属性有:偏移属性,段属性,类型属性。例如:

MOV BX, OFFSET VAL ;取偏移属性

MOV BX, SEG VAL ;取段属性

MOV BX, TYPE VAL ;取类型属性

6.3 指令和伪指令的区别在哪里?伪指令可以出现在代码段吗?指令可以在数据段吗?

答:指令只能出现在代码段,定义数据的伪指令通常在数据段,伪指令在代码段两端也可,但不能在指令之间。

6.4 下面的程序能否输出字符0~9?如不能,应如何修改?

CODE SEGMENT

ASSUME CS:CODE

K=30H

J DW 0

START: MOV DL, K

MOV AH, 2

INT 21H

K=K+1

INC J

CMP J, 10

JNZ START

MOV AH, 4CH

INT 21H

CODE ENDS

END START

答:通过汇编和连接可以运行,但程序真正意图是输出ASCII码为30H~39H的字符0~9,应作修改。

CODE SEGMENT

ASSUME CS:CODE

K=30H ;改为 K DB 30H

J DW 0

START: MOV DL, K

MOV AH, 2

INT 21H

K=K+1 ;K=K+1是伪指令,没生成代码,改为 INC K

INC J

CMP J, 10

JNZ START

MOV AH, 4CH

INT 21H

CODE ENDS

END START

6.5 用16位指令编写完整程序,并上机调试,计算V=(X+Y)*R,其中所有变量均为32位变量,X、Y、R的具体数值由你自己确定,变量定义格式如下:

X DW ?,?

Y DW ?,?

R DW ?,?

V DW 4 dup(?)

答:(32位指令)

data segment

x dw 1

y dw 2

r dw 3

v dw 4 dup(?)

data ends

code segment

assume cs:code,ds:data

.386p

start:

mov ax,data

mov ds,ax

mov eax,x

add eax,y

mov ebx,r

imul ebx

mov v,eax

mov v+4,edx

mov dl,al

add dl,30h

mov ah,2

int 21h

mov ah,4ch

int 21h

code ends

end start

6.6数据定义如下:执行下列指令,填写寄存器的值

ARRAY LABEL BYTE

DA1 DW 2,9,14,3

DA2 DB 7,‘ABCDEDF’

LEN = $-DA1

MOV AL,ARRAY+2 (AL)=( )H

ADD AL,DA2+1 (AL)=( )H

MOV AX,DA2-DA1 (AX)=( )H

MOV AX,DA1+1 (AX)=( )H

MOV BL,LEN (BL)=( )H

答: MOV AL,ARRAY+2 (AL)=( 09 )H

ADD AL,DA2+1 (AL)=( 41 )H

MOV AX,DA2-DA1 (AX)=( 0008 )H

MOV AX,DA1+1 (AX)=( 0900 )H

MOV BL,LEN (BL)=( 10 )H

6.7 定义数据段,满足如下要求:

(1)array为字符串变量:‘inspire a generation!’

(2)data1 为十六进制数:0FEDCBAH

(3)data2为二进制数:10101010B。

(4)data3为100个为零的字节变量。

(5)分配500个字的空间待用。

答: data segment

Array db ‘inspire a generation!’

Data1 df 0fedcbah

Data2 db 10101010B

Data3 db 100 dup(0)

dw 500 dup(?)

data ends

6.8 假设程序中,数据段定义如下:

Data1 db 50 dup(?)

Data2 dw 10 dup(0)

Data3 dq 5 dup(2 dup(1,2))

(1)用指令将数据段首地址放入数据段寄存器中。

(2)用一条指令将data2的第一个数据放入BX寄存器中。

(3)将数据段字节数放入CX寄存器。

答:mov ax,data1 mov ds,ax

Mov bx,data2[0]

Mov cx, 50+10*2+20*8

6.9现有数据定义如下:

Array1 dw 5 DUP(0)

Array2 EQU BYTE PTR Array1

请说明这两个变量之间的联系。

答:当汇编后,PTR类型操作符使Array2具有Array1相同的段地址和偏移地址,但它的数据类型为字节型。

6.10 给出下列程序段汇编后的结果:

Val1 EQU 6

Val2 EQU 3

MOV BX,(Val1 LT 5) AND 20

MOV BX, (VAL2 GE 1) AND 30

MOV BX,(Val2 AND 5) OR (VAL1 GE 5)

MOV BX,(Val2 – VAL1) GE 5

答:MOV BX,0 MOV BX,30 MOV BX,FF MOV BX,0

6.11设数据段定义如下:

Data segment

Org 20h

Data1=4

Data2=data1+25h

Data3 db ‘123456’

Db 47h,48h

Count EQU $-data1

Data4 dw 49h,4Ah,4Bh

Data ends

回答下列问题:

(1)Data1的偏移地址是多少?

(2)Count的值是多少?

答:(1)0020h (2)0024h

6.12现有一数据区data1,需对其进行按字和按字节访问,请问应如何进行设置?

答:可以利用“LABEL“标签进行设置,如:

Operator_b LABEL byte

Operator_w dw 100dup(0)

其中按字节访问时采用Operator_b变量,按字访问时采用Operator_w变量。

6.13请问什么是PSP,EXE文件和COM文件有何区别?

答:PSP是程序段前缀。程序在执行前调入内存,由DOS确定装入的起始地址,建立PSP,接着再装入程序,其大小为256个字节。EXE文件和COM文件相比,COM文件只有一个段地址,由二进制代码组成,比EXE文件小,并且要求程序从偏移地址0100H单元开始,因为之前存放为PSP。

习题7

7.1 下列程序是在3个数中找出最小的数并放入AL,在括号中填入指令使其完整。

  mov    al,x

mov bl,y

mov cl,z

cmp al,bl

( )

xchg al,bl

l1: cmp al, cl

jle l2

( )

l2: ret

答:(1)jl l1

(2)xchg al,cl

7.2 数据段如下:

data segment

da1 db 1,2,‘abcd’

count = $-da1

da2 db 9 dup(?)

data ends

补充括号处的指令,使得程序把DA1数据区数据移到DA2数据区。

mov ax,data

mov ds, ax

mov es,( )

mov cx,( )

mov si, ( )

( )

( )

答: mov es,( AX )

mov cx,( COUNT )

mov si, ( OFFSET DA1 )

( MOV DI, OFFSET DA2 )

( REP MOVSB )

7.3 将AX和BX进行加、减、乘或除的运算,每种运算由用户从键盘上选择。程序中设置寄存器的值,或在DEBUG下设定寄存器值并在DEBUG下运行程序。

答:code segment

assume cs:code

main proc far

push ds

sub ax,ax

push ax

mov ah,1

int 21h

cmp al,31h ;1加

jz ad

cmp al,32h ;2减

jz su

cmp al,33h ;3乘

jz mu

cmp al,34h ;4除

jz di

ad:ADD AX,BX

JMP exit

su:SUB AX,BX

JMP exit

mu:MUL BX

JMP exit

d:DIV BX

exit:

ret

main endp

code ends

end main

7.4 编写程序,从键盘接收一个小写字母,然后找出它的前导字符和后续字符,再按顺序显示这三个字母。

答:code segment

assume cs:code

main proc far

push ds

sub ax,ax

push ax

mov ah,1

int 21h

cmp al,61h

jb exit

cmp al,7ah

ja exit

mov bx,ax

sub al,1h ;输出前一字符

mov dl,al

mov ah,2

int 21h

mov dl,bl ;输出当前字符

mov ah,2

int 21h

add dl,1h ;输出后一字符

mov ah,2

int 21h

exit:ret

main endp

code ends

end main

7.5 分别用LOOP循环和条件转移指令实现1+2+3+……+100,并将结果存入AX。

答:(1)code segment

assume cs:code

main proc far

push ds

sub ax,ax

push ax

mov bx,1

mov ax,0

mov cx,99

a:add ax,bx

inc bx

loop a

exit:ret

code ends

end main

(2)code segment

assume cs:code

main proc far

push ds

sub ax,ax

push ax

mov bx,1

mov ax,0

mov cx,99

a:add ax,bx

inc bx

cmp bx,101

jnz a

exit:ret

main endp

code ends

end main

7.6 打印下面图形。

*

**





答:code segment

assume cs:code

main proc far

push ds

sub ax,ax

push ax

mov ax,1

a:mov cx,ax

b:mov ah,2

mov dl,’*’

int 21h

loop b

cmp ax,6

jz exit

inc ax

mov ah,2

mov dl,13

int 21h

mov ah,2

mov dl,10

int 21h

jmp a

exit:ret

code ends

end main

7.7 求已知带符号数字节数组ARRAY的平均值,ARRAY的首字节单元为数组元素的个数。

答:

data segment

ARRAY db 5,01,12,23,45,F3

data ends

code segment

assume cs:code

start:

mov ax,data

mov ds,ax

mov ax,0

mov cl,array

mov ch,0

mov bl,cl

lea si,array+1

a:add al,array[si]

inc si

loop a

div bl

exit:mov ah,4ch

int 21h

code ends

end start

7.8 编写程序,如果输入的是大写字母,则输出对应的小写字母;如果输入的是小写字母,则输出对应的大写字母;如果输入的是数字,原样输出;按回车结束。

答:code segment

assume cs:code

start:

mov ah,1

int 21h

cmp al,30h

jb exit

cmp al,3Ah

jb num

cmp al,41h

jb exit

cmp al,5Bh

jb large

cmp al,61h

jb exit

cmp al,7ah

ja exit

sub al,20h

jmp num

large: add al,20h

jmp num

num: mov dl,al

mov ah,2

int 21h

exit:mov ah,4ch

int 21h

code ends

end start

7.9 编写程序,实现对无符号字数组ARRAY的6个元素从小到大排序。

答: data segment

array dw 6,5,9,4,5,15,3 ;首地址单元6为元素个数

data ends

code segment

assume cs:code,ds:data

start:

mov ax, data

mov ds, ax

mov di, array

k1: mov cx, di

mov bx,0

k2: mov ax, array[bx]

cmp ax, array[bx+2]

jl next

xchg ax,array[bx+2]

mov array[bx],ax

next: add bx,2

loop k2

dec di

jnz k1

mov ah,4ch

int 21h

code ends

end start

7.10 数据段有两个等长的字数组,分别求出各自的元素之和,并存入元素后面的单元中,即横向相加。再求出两个数组的对应元素之和,并把和存入新数组SUM中,即纵向相加。

答:data segment

Array1 dw 3, 1,0,1,? ;设简单数据,第一个为元素个数

Array2 dw 3,1,1,0,?

Array3 dw 3,3 dup (0) ;存放array1,2的和

data ends

code segment

assume cs:code,ds:data

start:

mov ax, data

mov ds, ax

mov ax, 0

lea bx, array1+2

mov cx, array1

sumh1: add ax, [bx]

add bx,2

loop sumh1

mov [bx],ax

;第二个数组累加

mov ax, 0

lea bx, array2+2

mov cx, array2

sumh2: add ax, [bx]

add bx,2

loop sumh2

mov [bx],ax

lea bx,2

mov cx, array2

sum3: mov ax, array1[bx]

add ax, array2[bx]

mov array3[bx],ax

add bx,2

loop sum3

exit: mov ah,4ch

int 21h

code ends

end start

7.11 编写程序,比较两个从键盘输入的字符串是否相同,如果相同,则显示 ‘YES’,如果不同,则显示发现不同的字符位置。

答:data segment

mess1 db 13,10, ‘input string1:$’

mess2 db 13,10, ‘input string2:$’

mess3 db 13,10, ‘YES!$’

mess4 db 13,10, ‘no match at $’

st1 label byte

max1 db 6

act1 db ?

stok1 db 6 dup(?)

st2 label byte

max2 db 6

act2 db ?

stok2 db 6 dup(?)

data ends

code segment

assume cs:code, ds:data,es:data

start: mov ax,data

mov ds,ax

mov es,ax

lea dx,mess1

mov ah,09

int 21h ;qust1?

lea dx,st1

mov ah,0ah

int 21h ;ans1

lea dx,mess2

mov ah,09

int 21h ;qust2?

lea dx,st2

mov ah,0ah

int 21h ;ans2

mov cl,act1

mov dl,act2

cmp cl,dl

jnz nomatch

mov ch,0

lea si,stok1

lea di,stok2

repe cmpsb

jnz nomatch

match: lea dx,mess3

mov ah,09

int 21h

jmp exit

nomatch:

lea dx,mess4

mov ah,09

int 21h

sub di,offset stok2

mov dx,di

add dl,30h

mov ah,2

int 21h

exit:

mov ah,4ch

int 21h

code ends

end start

7.12 编写程序,从键盘输入一个字符串到BUFF,再输入一个字符到AL,在字符串BUFF中查找是否存在该字符,如果找到,显示发现的字符位置。

答:

data segment

mess1 db 13,10, ‘input string:$’

mess2 db 13,10, ‘input a char:$’

mess3 db 13,10, ‘found at $’

mess4 db 13,10, ‘no found !$’

st1 label byte

max1 db 6

act1 db ?

stok1 db 6 dup(?)

data ends

code segment para’code’

assume cs:code, ds:data

start: mov ax,data

mov ds,ax

mov es,ax

lea dx,mess1

mov ah,09

int 21h ;qust1?

lea dx,st1

mov ah,0ah

int 21h ;ans1

lea dx,mess2

mov ah,09

int 21h ;qust2?

mov ah,1

int 21h ;ans2

lea di,stok1

repne scasb

jz match

nomatch:

lea dx,mess4

jmp exit

match: lea dx,mess3

exit:

mov ah,09

int 21h

sub di,offset stok1

mov dx,di

and dx,0fh

add dl,30h

mov ah,2

int 21h

mov ah,4ch

int 21h

code ends

end start

7.13 编写程序,从键盘输入一个字符串到BUFF,并按相反顺序显示输出。

答:

data segment

mess1 db 13,10, ‘input string:$’

mess2 db 13,10,’$’

st1 label byte

max1 db 6

act1 db ?

stok1 db 6 dup(?)

data ends

code segment

assume cs:code, ds:data

start : mov ax,data

mov ds,ax

lea dx,mess1

mov ah,09

int 21h

lea dx,st1

mov ah,0ah

int 21h

lea dx,mess2

mov ah,09

int 21h

mov cl,act1

mov ch,0

mov bx, offset stok1

add bx, cx

next: dec bx

mov dl, [bx]

mov ah,2

int 21h

loop next

mov ah,4ch

int 21h

code ends

end start

7.14 编写程序,从键盘输入一个八位的二进制数,显示其十六进制数。

答:code segment

assume cs:code

start:

mov cx,8

mov bl,0

next: mov ah,1

int 21h

cmp al,30h

jb exit

cmp al,31h

ja exit

sub al,30h

shl bl,1

add bl,al

loop next

mov cl,4

mov di,2

out1: rol bl,cl

mov dl,bl

and dl,0fh

add dl,30h

cmp dl,39h

jle dig

add dl,7

dig: mov ah,2

int 21h

dec di

cmp di,0

jnz out1

exit: mov ah,4ch

int 21h

code ends

end start

7.15 字数组ARRAY为有符号数,第一个单元为元素个数N,后面为N个元素,编写程序,求数组元素中的最大值,并把它放入MAX单元。

答:data segment

array dw 5,9,4,5,15,3

max dw ?

data ends

code segment

assume cs:code,ds:data

start:

mov ax, data

mov ds, ax

mov cx, array

lea bx, array+2

mov ax, [bx]

mov max, ax

k1: mov ax, [bx]

cmp ax, max

jl next

mov max,ax

next: add bx,2

loop k1

mov ah,4ch

int 21h

code ends

end start

7.16 字数组ARRAY,第一个单元为元素个数N,后面为N个元素,编写程序,把零元素从数组中清除,移动元素位置并修改第一个单元(元素个数)。

答:data segment

array dw 6,3,4,0,15,0,7

loca dw 999 ;此单元地址为结束标记

data ends

code segment

assume cs:code,ds:data

start:

mov ax, data

mov ds, ax

mov cx, array

lea bx, array+2

next: cmp word ptr[bx],0

jz move

add bx,2

dec cx

cmp cx,0

jnz next

jmp exit

move: dec array

mov di,bx

mnext:

cmp di,offset loca

ja next

mov ax, [di+2]

mov [di],ax

add di,2

cmp di,offset loca

jnb next

jmp mnext

exit: mov ah,4ch

int 21h

code ends

end start

习题8

8.1 过程定义如下,补充括号中的指令。

code segment

assume cs:code

main proc far

( )

( )

( )

ret

main endp

code ends

end main

答:( PUSH DS )

( XOR AX,AX )

( PUSH AX )

8.2 补充下列程序括号中的指令,使得程序对堆栈的操作全部利用程序中定义的TOS堆栈,并画出程序执行后堆栈TOS中的数据。

data segment

dw 100 dup(?)

tos label word

data ends

code segment

assume cs:code,ss:data

main proc far

( )

( )

( )

push ds

xor ax, ax

push ax

call far ptr suba

… ;假定此处指令的地址为CS=3400h,IP=30h

code ends

end main

答:( MOV AX,DATA )

( MOV SS,AX )

( LEA SP,TOS )

8.3 主程序从键盘输入一个字符串到BUFF,再输入一个字符到AL,用子程序在字符串BUFF中查找是否存在该字符,如果找到,显示发现的字符位置。用寄存器传递要查找的字符。

答:Data segment

BUFF1 db 16,?,16 dup(?),13,10,’$’

Data ends

Code segment

Assume cs:code,ds:data

Main proc far

Push ds

Xor ax,ax

Push ax

Mov ax,data

Mov ds,ax

Mov es,ax

Lea dx,BUFF1

Mov ah,10

Int 21h

mov ah,1

int 21h

Lea di,BUFF1+2

Mov cl,BUFF1+1

mov ch,0

Mov ah,0

Repne scasb

Jnz ye

dec di

mov bx,di

Call BTH

Ye: ret

Main endp

BTH proc near

mov cx,4

s:rol bx,1

rol bx,1

rol bx,1

rol bx,1

mov al,bl

and al,0fh

add al,30h

cmp al,39h

jle d

add al,7

d:mov dl,al

mov ah,2

int 21h

loop s

ret

BTH endp

Code ends

End main

8.4 主程序从键盘输入一个八位的二进制数,对其作求补码操作,用子程序对求补后的值以二进制形式显示。(正数的补码=输入)

答:Code segment

Assume cs:code

Main proc far

Push ds

Xor ax,ax

Push ax

Mov bx,0

Mov cx,8

A:mov ah,1

Int 21h

Shl bx,1

Sub al,30h

Jz b ;输入为0

INC bx ;输入为1

B:loop a

Test bx,0080h

Jz d

XOR bx,00FFh ;取反

INC bx

D:Call disp

Ret

Main endp

disp proc near

mov cx,8

shl bx,cl

ls1:shl bx,1

jnc k30

mov dl,31h

jmp outb

k30: mov dl,30h

outb:mov ah,2

int 21h

loop ls1

ret

disp endp

code ends

end main

8.5 主程序从键盘(连续)输入两个四位的十六进制数A和B,用子程序作十六进制计算A+B,并显示计算结果(二进制)。

答:

data segment

A dw ?

B dw ?

data ends

Code segment

Assume cs:code,ds:data

Main proc far

Push ds

Xor ax,ax

Push ax

Mov ax,data

Mov ds,ax

Call INH ;输入十六进制数到BX寄存器

Mov A,bx

Call INH ;输入十六进制数到BX寄存器

Mov B,bx

Call ad

Ret

Main endp

INH proc near

mov bx,0 ;初始化

mov ch,4

mov cl,4

inchr: mov ah,1 ;键盘输入

int 21h

cmp al,30h

jl exit ;非法输入

cmp al,39h

jle dig ;输入是数字0~9

cmp al,41h

jl exit ;非法输入

cmp al,46h

jg exit ;非法输入

sub al,37h ;输入是大写a~f

jmp ls4

dig: sub al,30h

ls4: shl bx,cl

add bl,al

dec ch

jnz inchr

exit: ret

INH endp

Ad proc near

Mov ax,a

Mov bx,b

Add bx,ax

Call disp

Ret

Ad endp

disp proc near

mov cx,16

ls1: shl bx,1

jnc l30

mov dl,31h

jmp outb

l30; mov dl,30h

outb: mov ah,2

int 21h

loop ls1

ret

disp endp

Code ends

End main

8.6 某字数组为有符号数,第一个单元为元素个数N,后面为N个元素,编写通用子程序,求数组元素中的最大值,并把它放入MAX单元。

答:MAX_p proc near

Mov di,[bx] ;个数地址

Mov si,[bx+2] ;数组地址

Mov cx,[di]

Mov di,MAX

Xor ax,ax

Next:cmp di,[si]

Jg a

Mov di,[si]

A: Add si,2

Loop next

Mov MAX,di

ret

MAX_p endp

8.7 设有一个数组存放学生的成绩(0 ~ 100),编制一个子程序统计0 ~ 59分、60 ~ 69分、70 ~ 79分、80 ~ 89分、90 ~ 100分的人数,并分别存放到scoreE、scoreD、score C、score B及score A单元中。编写一个主程序与之配合使用。

答:

data segment

Array db 6,9,65,78,68,86,93 ;6为学生人数,后面为6个成绩

ScoreE db ?

scoreD db ?

scoreC db ?

scoreB db ?

scoreA db ?

Data ends

Code segment

Assume cs:code,ds:data

Main proc far

Mov ax,data

Mov ds,ax

Mov cl,array

mov ch,0

Call class

Ret

Main endp

Class proc near

Lea si, array

next:

inc si

Mov bl, [si]

Cmp bl,60

Jl e

Cmp bl,70

Jl d

Cmp bl,80

Jl k

Cmp bl,90

Jl b

Mov dl, scoreA

Inc dl

Mov scoreA,dl

loop next

jmp exit

B: Mov dl, scoreB

Inc dl

Mov scoreB,dl

loop next

jmp exit

k: Mov dl, scoreC

Inc dl

Mov scoreC,dl

loop next

jmp exit

D: Mov dl, scoreD

Inc dl

Mov scoreD,dl

loop next

jmp exit

E: Mov dl, scoreE

Inc dl

Mov scoreE,dl

loop next

jmp exit

exit:ret

class endp

Code ends

End main

8.8 用多模块程序设计一个简单的计算器程序,实现整数的加减乘除。运算符可以为:+,-,*,/,=。

答:

public num1,num2,res

extrn ad:far,su:far,mu:far,dv:far,el:far

data segment

num1 db ?

num2 db ?

res dw ?

data ends

code segment

assume cs:code,ds:data

main proc far

push ds

sub ax,ax

push ax

mov ax,data

mov ds,ax

call DTOB

mov num1,bx

call DTOB

mov num2,bx

mov ah,1

int 21h

cmp dl,’+’

jz a

cmp dl,’-’

jz s

cmp dl,’*’

jz m

cmp dl,’/’

jz d

cmp dl,’=’

call el

jmp exit

a:call ad

jmp exit

s:call su

jmp exit

m:call mu

jmp exit

d:call dv

exit:ret

main endp

dtob proc near

mov bx, 0

input: mov ah, 1 ;键盘输入

int 21h

sub al, 30h ;把ascii码转变为数值

jl exit ;如不是数则退出

cmp al, 9

jg exit ;如不是数则退出

cbw ;扩展为字

xchg ax, bx ;交换寄存器

mov cx, 10

mul cx ;a(n)= a(n-1)×10

xchg ax, bx ;交换寄存器

add bx, ax ;a(n)=a(n)+b(n)

jmp input

exit: ret

dtob endp

code ends

end main

;——————

public ad

extrn num1:byte,num2:byte,res:word

code segment

assume cs:code

ad proc near

mov ah,0

mov al,num1

add al,num2

adc ah,0

mov res,ax

ret

order endp

su proc near

mov ah,0

mov al,num1

sub al,num2

sbb ah,0

mov res,ax

ret

su endp

mu proc near

mov al,num1

mul num2

mov res,ax

ret

mu endp

dv proc near

mov al,num1

div num2

mov res,ax

ret

dv endp

el proc near

mov al,num1

cmp al,num2

jz z

mov res,0FFFFH ;不相等

jmp exit

z: mov res,0 ;相等

exit:ret

el endp

code ends

end

8.9 从键盘输入姓名和电话号码,建立通讯录,通讯录的最大容量为9条记录,程序结束时无须保留通讯录,但程序运行时要保留通讯录信息。程序的人机界面和顺序要求如下:

(1)提示信息INPUT NAME:(调用子程序INNAME录入姓名,序号自动产生)

(2)提示信息INPUT TELEPHONE NUMBER:(调用子程序INTELE录入电话号码)

(3)提示信息INPUT 序号:(调用子程序PRINT显示某人的姓名和电话号码,如果序号不存在,则提示信息 NO THIS NUMB )。

答:Inform struc

Sn db ?

Name db ‘??????’

Telen db ‘???????????’

Inform ends

Data segment

Array struc 9 dup(<>)

Name db ‘INPUT NAME $’

TEL db ‘INPUT TELEPHONE NUMBER $’

number db ‘INPUT SEQUENCE NUMBER $’

err db ‘NO THIS NUMB $’

Data ends

code segment

assume cs:code,ds:data

main proc far

push ds

xor ax,ax

push ax

mov ax,data

mov ds,ax

mov cx,0

a:

call inname

call intele

inc cx

cmp cx,9

jl a

call print

ret

main endp

inname proc near

push cx

lea dx,name

mov ah,9

int 21h

mov al, size array

mul cl

mov bx,ax

add bx,array

mov [bx].sn,cl

mov di,0

a:mov ah,1

int 21h

mov [bx].name[di],al

inc di

cmp di,6

jl a

pop cx

ret

inname endp

intele proc near

push cx

lea dx,tel

mov ah,9

int 21h

mov al, size array

mul cl

mov bx,ax

add bx,array

mov di,0

a:mov ah,1

int 21h

mov [bx].telen[di],al

inc di

cmp di,13

jl a

pop cx

ret

intele endp

print proc near

lea dx,number

mov ah,9

int 21h

mov ah,1

int 21h

cmp al,0

jl ex

cmp al,8

jg ex

mov cl,al

mov al, size array

mul cl

mov bx,ax

add bx,array

mov dx,bx

mov ah,9

int 21h

mov dl,13

mov ah,2

int 21h

mov dl,10

mov ah,2

int 21h

lea dx,[bx].telen

mov ah,9

int 21h

jmp r

ex: lea dx,err

mov ah,9

int 21h

r:

ret

print endp

code ends

end main

习题9

9.1 宏定义:

MSG MACRO P1,P2,P3

IN&P1 P2 P3

ENDM

K=1

展开下列宏调用:

MSG %K,DB,‘MY NAME’

MSG C, AX

答: 1 IN1 DB ‘MY NAME’

2 INC AX

9.2 使用宏指令,在数据段定义九条通讯录记录,宏展开后的数据段形如:

DATA SEGMENT

DA1 LABEL BYTE

DB 1,‘NAME1’,‘TELE1’

DB 2,‘NAME2’,‘TELE2’

DB 9,‘NAME9’,‘TELE9’

DATA ENDS

答: DEF MACRO

IRP X,<1,2,3,4,5,6,7,8,9>

DB X,’NAME&X&’,’TELE&X&’

ENDM

DATA SEGMENT

DA1 LABEL BYTE

DEF

DATA SEGMENT

9.3 宏指令和指令的区别是什么?使用宏指令和使用子程序有何异同?宏指令的优点在哪里?

答:宏指令只在汇编时起作用,指令汇编后继续执行。宏指令与子程序的区别如下:(1)空间的区别:宏指令大于子程序,多次调用宏指令,程序长度增加。

(2)时间的区别:宏运行不需要额外的时间,子程序需要。

(3)参数的区别:宏命令可实现多个参数的直接代换,方式简单灵活;而子程序参数传递麻烦。

总之,代码不长和变元较多的功能段,使用宏命令比较合适。

9.4 在宏定义中有时需要LOCAL 伪操作,为什么?

答:在宏定义中,常常使用标号,当多次宏调用后,就会出现标号重复定义的错误。使用LOCAL伪操作,对标号说明为局部标号,这样每次调用,宏展开的标号是不同的。

9.5 宏定义在程序中的位置有何规定?宏调用是否一定放在代码段?

答:宏定义在程序中的位置没有严格要求,可以写在某一段内,也可以不在段内。

9.6 用宏指令计算S=(A+B)*K/2,其中A,B,K为常量。

答: math macro S,A,B,K

S=(A+B)*K/2

ENDM

9.7 编写宏定义,比较两个常量X和Y,如果X>Y,MAX=X,否则MAX=Y。

答:COMPARE MACRO MAX,X,Y

LOCAL A,B

CMP X,Y

JGE A

MOV MAX,Y

JMP B

A: MOV MAX,X

B:

ENDM

9.8 编写非递归的宏定义,计算K的阶乘,K为变元。

答:FAC MACRO K

MOV AX,1

MOV CX,K+1

A:SUB CX,1

MUL CX

CMP CX,1

JG A

ENDM

9.9 在数据段中定义了三个有符号数A、B、C,使用宏指令,给三个数排序,三个变量作为参数。

答:ORDER MACRO A,B,C

CMP A,B

JG L1

MOV AX,B

MOV BX,A

MOV A,AX

MOV B,BX

L1:CMP B,C

JG L2

MOV AX,B

MOV BX,C

MOV C,AX

CMP A,BX

JG L2

MOV AX,A

MOV A,BX

MOV B,AX

L2:

ENDM

9.10 编写一个宏定义SCAN,完成在一个字符串中查找某个字符的工作。被查找的该字符,字符串首地址及其长度均为变元。

答: SCAN MACRO A,STREE,N

MOV AX,SEG STREE

MOV ES,AX

LEA DI,STREE

MOV CX,N

MOV AL,A

CLD

REPNE SCASB

JZ YES

MOV DL,’N’

JMP DISP

YES:MOV DL,’Y’

DISP:MOV AH,2

INT 21H

ENDM

9.11 编写宏指令COMPSS,比较2个同长度的字符串str1和str2是否相等, 2个字符串的首地址和长度为变元。写出完整程序,在数据段中写出数据定义,在代码段中写出宏定义和宏调用。并处理若相等则显示‘MATCH’,否则显示‘NOT MATCH’.

答:

COMPSS MACRO ST1,ST2,N

LEA SI,ST1

LEA DI,ST2

MOV CX,N

CLD

REPE CMPSB

JZ YES

MOV DX,LEA S2

JMP DISP

YES:MOV DX,LEA S1

DISP:MOV AH,9

INT 21H

ENDM

DATA SEGMENT

STR1 DB ‘’

NUM DB ?

S1 DB ‘MATCH$’

S2 DB ‘NOT MATCH$’

DATA ENDS

EXT SEGMENT

STR2 DB ‘’

EXT ENDS

CODE SEGMENT

MAIN PROC FAR

PUSH DS

PUSH ES

MOV AX,0

PUSH AX

MOV AX,DATA

MOV DS,AX

MOV AX,EXT

MOV ES,AX

MOV NUM, LENGTH STR1

COMPSS STR1,STR2, NUM

RET

CODE ENDS

END MAIN

9.12 编写宏定义程序,可以对任意字数组求元素之和,数组名称、元素个数和结果存放单元为宏定义的哑元。

答:ADS MACRO ARRAY,N,SUM

Lea di,ARRAY

Mov cx,N

Mov ax,0

Mov si,0

Ad:add ax,[di][si]

Add si,2

Loop ad

Mov SUM,ax

ENDM

9.13 编写一个宏库文件,其中包括系统功能调用(INT 21H)的00~0A号功能调用。并通过宏调用实现以下各项功能:从键盘输入一个字符串到BUFF;再输入一个单字符,然后在字符串BUFF中查找是否存在该字符;如果找到,显示发现的字符位置。

答: 宏库文件sysmacro.mac文件定义如下:

INPUT MACRO

MOV AH,1

INT 21H

ENDM

OUTPUT MACRO CHA

MOV DL,CHA

MOV AH,2

INT 21H

ENDM

INPUTN MACRO

MOV AH,7

INT 21H

ENDM

OUTSTR MACRO STREE

MOV DX,OFFSET STREE

MOV AH,9

INT 21H

ENDM

INBUFF MACRO BUFF

MOV DX,OFFSET BUFF

MOV AH,0AH

INT 21H

ENDM

WFILE MACRO FIL,NUM,COUNT

MOV DX,OFFSET FIL

MOV BX,NUM

MOV CX,COUNT

MOV AH,40H

INT 21H

ENDM

EXIT MACRO

MOV AH,4CH

INT 21H

ENDM

调用:include sysmacro.mac

Data segment

BUFF1 db 16,?,16 dup(?),13,10,’$’

Data ends

Code segment

Assume cs:code,ds:data

Main proc far

Push ds

Xor ax,ax

Push ax

Mov ax,data

Mov ds,ax

Mov es,ax

INBUFF BUFF1

INPUT

Lea di,BUFF1+2

Mov cl,BUFF1+1

mov ch,0

Mov ah,0

Repne scasb

Jnz ye

dec di

mov bx,di

Call BTH

Ye: ret

Main endp

BTH proc near

mov cx,4

s:rol bx,1

rol bx,1

rol bx,1

rol bx,1

mov al,bl

and al,0fh

add al,30h

cmp al,39h

jle d

add al,7

d:OUTPUT al

loop s

ret

BTH endp

Code ends

End main

习题10

10.1 I/O数据传送控制方式有哪几种?

答:无条件传送、查询传送、中断传送、直接存储器传送。

10.2 什么是I/O接口?什么是端口?接口部件在计算机主机一方还是在外设一方?

答:I/O接口是INPUT/OUTPUT指输入/输出设备接口。CPU与外部设备的连接和数据交换都需要通过接口设备来实现,被称为I/O接口。计算机主机的CPU通过访问这些寄存器来实现与外部设备交换数据。把接口中的这些寄存器称为端口(PORT)。这些寄存器并不是在外部设备一方,而是属于计算机主机。

10.3 通过端口传递哪三种信息?

答:CPU与I/O设备的通信有三种信息,即控制信息、状态信息和数据信息。

10.4 根据以下要求写出输入输出指令:

(1) 读61H端口 (2) 写20H端口

(3) 读3F8H端口 (4) 写3F9H端口

答:(1) IN al,61h

(2) OUT 20h,al

(3)mov dx,3F8H IN al,dx

(4)mov dx,3F9H OUT dx,al

10.5 举例说明何为中断类型号,何为中断向量,何为中断向量表。对于INT 8指令,中断向量存放的内存地址是多少?

答:在实际的系统中,中断源有多个,需要给每个中断源编一个号,以便于识别。在执行软件中断指令INT n时,n就是中断类型号。我们把中断处理程序的起始地址称为中断向量。在存储器的最低1KB(地址从0000~3FFH)集中存放256种中断类型的中断向量,每个中断向量为四个字节,其中前两个字节是偏移地址,后两个字节是段地址。这个集中存放中断向量的存储区称为中断向量表。

INT 8对应中断向量存放的内存地址是00020H

10.6 举例说明何为内中断,何为外中断,何为硬件中断,何为可屏蔽中断。

答:内中断:中断指令INT

外中断:也称为硬件中断,如电源故障。

可屏蔽中断:如定时器。

10.7 何为开中断和关中断。关中断情况下,内中断能否被响应?不可屏蔽中断能否被响应?可屏蔽中断能否被响应?

答:所谓开中断,是指程序在运行中,包括中断程序执行中,允许响应其他中断请求,再转去运行中断服务程序。关中断则是不允许系统打断连续的运行。内中断可以响应,不可屏蔽中断可以被相应。可屏蔽中断不能被响应。

10.8 写出仅使定时器、键盘、硬盘、打印机不被屏蔽,并开中断的指令序列。

答: Mov al,01011100

OUT 21h,al

STI

10.9 列出INT指令执行的操作,列出CPU响应外部中断时所做的工作。

答:中断请求,中断优先级判定,中断响应,中断处理,中断返回。

CPU响应外部中断自动完成下列工作:

1.取中断内型号N;

2.标志寄存器入栈;

3.代码段寄存器和指令指针入栈;

4.禁止硬件中断和单步中断;

5.在中断向量表中的N*4开始的单元取两个字分别送IP和CS,获取

中断处理程序入口地址。

10.10 中断处理程序中通常要做哪些工作?中断处理程序中是否一定要开中断?如果有开中断指令,意味着什么?

答:中断处理程序一开始可以根据需要开中断,以允许中断嵌套。用入栈指令把中断处理程序中将要用到的寄存器内容压入堆栈,以保护现场,待中断处理完毕,退出中断处理程序之前再把寄存器的内容从堆栈中弹出,从而恢复现场。

10.11 为什么说程序员应尽可能使用层次较高的DOS功能,其次才是使用BIOS功能,最后才是使用输入输出指令。

答:因为用户通过调用DOS或者BIOS例行程序来实现对外设的访问,可以降低程序设计的复杂程度,缩短开发周期。BIOS存放在机器的ROM中,层次比DOS低,更接近硬件;DOS功能调用是操作系统DOS的一个组成部分,开机时从外存装入内存,DOS可以完成比BIOS更高级的功能,因此DOS操作比使用相应功能的BIOS操作更简易,而且DOS对硬件的依赖性更少。

10.12 编写程序段,使用INT 1CH指令调用首地址为MYINT的中断处理程序。用完后恢复INT 1CH指令的原有功能。

Code segment

Assume cs:code

Start:

Mov al,1ch

Mov ah,35h

Int 21h

Push es

Push bx

Push ds

Mov dx,offset MYINT

Mov ax,seg MYINT

Mov ds,ax

Mov al,1ch

Mov ah,25h

Int 21h

;

;使用中断程序

;

Pop dx

Pop ds

Mov al,1ch

Mov ah,25h

Int 21h

Mov ah,4ch

Int 21h

Code ends

End start

习题11 (基本可在书中找到参考,省略)

;