《汇编语言程序设计》教案 附:习题参考答案
《IBM-PC汇编语言程序设计》 (第2版)
沈美明、温冬婵 编著
教案编写时间:2007年8月18日
目录
第 1 章 基础知识 ...................................................................................................................................................... 1 1.1 进位计数制与不同基数的数之间的转换 .....................................................................................................1 1.2 二进制数和十六进制数的运算 .....................................................................................................................2 1.3 计算机中数和字符的表示 .............................................................................................................................3 1.4 几种基本的逻辑运算 .....................................................................................................................................3 第 2 章 80X86计算机组织 ...................................................................................................................................... 4 2.1 80X86微处理器 ...............................................................................................................................................4 2.2 基于微处理器的计算机系统构成 .................................................................................................................4 2.3 中央处理机 .....................................................................................................................................................5 2.4 存储器 .............................................................................................................................................................6 2.5 外部设备 .........................................................................................................................................................7 第 3 章 80X86的指令系统和寻址方式 .................................................................................................................. 8 3.1 80X86的寻址方式 ...........................................................................................................................................8 3.2 程序占有的空间和执行时间 ....................................................................................................................... 10 3.3 80X86的指令系统 ......................................................................................................................................... 10 第 4 章 汇编语言程序格式 .................................................................................................................................... 26 4.1 汇编程序功能 ............................................................................................................................................... 26 4.2 伪操作 ........................................................................................................................................................... 26 4.3 汇编语言程序格式 ....................................................................................................................................... 30 4.4 汇编语言程序的上机过程 ........................................................................................................................... 33 第 5 章 循环与分支程序设计 ................................................................................................................................ 35 5.1 循环程序设计 ............................................................................................................................................... 35 5.2 分支程序设计 ............................................................................................................................................... 36 5.3 如何在实模式下发挥80386及其后继机型的优势 .................................................................................... 36 第 6 章 子程序结构 ................................................................................................................................................ 37 6.1 子程序的设计方法 ....................................................................................................................................... 37 6.2 子程序的嵌套 ............................................................................................................................................... 38 6.3 子程序举例 ................................................................................................................................................... 38 第 7 章 高级汇编语言技术 .................................................................................................................................... 39 7.1 宏 汇 编 ........................................................................................................................................................ 39 7.2 重 复 汇 编 .................................................................................................................................................. 40 7.3 条 件 汇 编 .................................................................................................................................................. 41 第 8 章 输入/输出程序设计 ................................................................................................................................... 42 8.1 I/O设备的数据传送方式 .............................................................................................................................. 42 8.2 程序直接控制I/O方式 ................................................................................................................................ 43 8.3 中断传送方式 ............................................................................................................................................... 43 第 9 章 BIOS和DOS中断 ................................................................................................................................... 46 9.1 键盘I/O ......................................................................................................................................................... 46 9.2 显示器I/O ..................................................................................................................................................... 48 9.3 打印机I/O ..................................................................................................................................................... 49 9.4 串行通信口I/O ............................................................................................................................................. 50 第 10 章 图形与发声系统的程序设计 .................................................................................................................. 51 10.1 显示方式 ..................................................................................................................................................... 51
10.2 视频显示存储器 ......................................................................................................................................... 51 10.3 EGA/VGA图形程序设计 ........................................................................................................................... 52 10.4 通用发声程序 ............................................................................................................................................. 53 10.5 乐曲程序 ..................................................................................................................................................... 54 第 11 章 磁盘文件存取技术 .................................................................................................................................. 55 11.1 磁盘的记录方式 ......................................................................................................................................... 55 11.2 文件代号式磁盘存取 ................................................................................................................................. 56 11.3 字符设备的文件代号式I/O ....................................................................................................................... 57 11.4 BIOS磁盘存取功能 .................................................................................................................................... 58 附录:《IBM—PC汇编语言程序设计》习题参考答案 ................................................................................. 59
第 一 章.
第 二 章. 第 三 章. 第 四 章. 第 五 章. 第 六 章. 第 七 章. 第 八 章. 第 九 章. 第 十 章. 第 十一 章. 习 题 ........................................................................................................................ 59 习 题 ........................................................................................................................ 60 习 题 ........................................................................................................................ 61 习 题 ........................................................................................................................ 74 习 题 ........................................................................................................................ 79 习 题 ........................................................................................................................ 97 习 题 ...................................................................................................................... 110 习 题 ...................................................................................................................... 117 习 题 ...................................................................................................................... 122 习 题 ...................................................................................................................... 125 习 题 ...................................................................................................................... 136
附录:《IBM—PC汇编语言程序设计》习题参考答案
第 一 章. 习 题
1.1 用降幂法和除法将下列十进制数转换为二进制数和十六进制数:
(1) 369 (2) 10000 (3) 4095 答:(1) 369=1 0111 0001B=171H
(2) 10000=10 0111 0001 0000B=2710H (3) 4095=1111 1111 1111B=FFFH
(4) 32767=111 1111 1111 1111B=7FFFH 1.2 将下列二进制数转换为十六进制数和十进制数:
(1) 10 1101 (2) 1000 0000 答:(1) 10 1101B=2DH=45
(2) 1000 0000B=80H=128
(3) 1111 1111 1111 1111B=FFFFH=65535 (4) 1111 1111B=FFH=255
(4) 32767
(3) 1111 1111 1111 1111 (4) 1111 1111
1.3 将下列十六进制数转换为二进制数和十进制数:
(1) FA (2) 5B (3) FFFE 答:(1) FAH=1111 1010B=250
(2) 5BH=101 1011B=91
(3) FFFEH=1111 1111 1111 1110B=65534 (4) 1234H=1 0010 0011 0100B=4660
(4) 1234
1.4 完成下列十六进制数的运算,并转换为十进制数进行校核:
(1) 3A+B7 (2) 1234+AF (3) ABCD-FE 答:(1) 3A+B7H=F1H=241
(2) 1234+AFH=12E3H=4835 (3) ABCD-FEH=AACFH=43727 (4) 7AB×6FH=35325H=217893
(4) 7AB×6F
1.5 下列各数均为十进制数,请用8位二进制补码计算下列各题,并用十六进制数表示其运算结果。
(1) (-85)+76 (2) 85+(-76) (3) 85-76 (4) 85-(-76) (5) (-85)-76 (6) -85-(-76)
答:(1) (-85)+76=1010 1011B+0100 1100B=1111 0111B=0F7H;CF=0;OF=0
(2) 85+(-76)=0101 0101B+1011 0100B=0000 1001B=09H;CF=1;OF=0
(3) 85-76=0101 0101B-0100 1100B=0101 0101B+1011 0100B=0000 1001B=09H;CF=0;OF=0 (4) 85-(-76)=0101 0101B-1011 0100B=0101 0101B+0100 1100B=10100001B=0A1H;CF=0;OF=1 (5) (-85)-76=1010 1011B-0100 1100B=1010 1011B+1011 0100B=0101 1111B=5FH;CF=0;OF=1 (6) -85-(-76)=1010 1011B-1011 0100B=1010 1011B+0100 1100B=11110111B=0F7H;CF=0;OF=0 1.6 下列各数为十六进制表示的8位二进制数,请说明当它们分别被看作是用补码表示的带符号数或
无符号数时,它们所表示的十进制数是什么? (1) D8 (2) FF
答:(1) D8H表示的带符号数为 -40,D8H表示的无符号数为216;
(2) FFH表示的带符号数为 -1, FFH表示的无符号数为255。 1.7 下列各数均为用十六进制表示的8位二进制数,请说明当它们分别被看作是用补码表示的数或字
符的ASCII码时,它们所表示的十进制数及字符是什么?
(1) 4F (2) 2B (3) 73 答:(1) 4FH表示的十进制数为 79,4FH表示的字符为O;
(2) 2BH表示的十进制数为 43,2BH表示的字符为 +; (3) 73H表示的十进制数为115,73H表示的字符为s; (4) 59H表示的十进制数为89,59H表示的字符为Y。
(4) 59
1.8 请写出下列字符串的ASCII码值。
For example,
This is a number 3692.
答:46H 6FH 72H 20H 65H 78H 61H 6DH 70H 6CH 65H 2CH 0AH 0DH
54H 68H 69H 73H 20H 69H 73H 20H 61H 20H 6EH 75H 6DH 62H 65H 72H 20H 33H 36H 39H 32H 2EH 0AH 0DH
第 二 章. 习 题
2.1 在80x86微机的输入/输出指令中,I/O端口号通常是由DX寄存器提供的,但有时也可以在指令中
直接指定00~FFH的端口号。试问可直接由指令指定的I/O端口数。 存储器 答:可直接由指令指定的I/O端口数为256个。 2.2 有两个16位字1EE5H和2A3CH分别存放在80x86微机的存储器的000B0H
和000B3H单元中,请用图表示出它们在存储器里的存放情况。 答:存储器里的存放情况如右上图所示。 2.3 在IBM PC机的存储器中存放信息如右下图所示。试读出30022H和30024H
字节单元的内容,以及30021H和30022H字单元的内容。 答:30022H字节单元的内容为ABH;30024H字节单元的内容为EFH。30021H
字单元的内容为AB34H;30022H字单元的内容为CDABH。 2.4 在实模式下,段地址和偏移地址为3017:000A的存储单元的物理地址是
什么?如果段地址和偏移地址是3015:002A和3010:007A呢?
答:3017:000A、3015:002A和3010:007A的存储单元的物理地址都是
3017AH。 2.5 如果在一个程序开始执行以前(CS)=0A7F0H,(如16进制数的最高位为字
母,则应在其前加一个0) (IP)=2B40H,试问该程序的第一个字的物理地址是多少?
答:该程序的第一个字的物理地址是0AAA40H。
000B0H 000B1H 000B2H 000B3H 000B4H E5H 1EH … 3CH 2AH
2.2题的信息存放情况
存储器 30020H 30021H 30022H 30023H 30024H
12H 34H ABH CDH EFH
2.3题的信息存放情况
2.6 在实模式下,存储器中每一段最多可有10000H个字节。如果用调试程序DEBUG的r命令在终端上
显示出当前各寄存器的内容如下,请画出此时存储器分段的示意图,以及条件标志OF、SF、ZF、CF的值。
10E40
C>debug 10F40 数据段 -r 20E40 附加段 20F40 AX=0000 BX=0000 CX=0079 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000 DS=10E4 ES=10F4 SS=21F0 CS=31FF
21F00 IP=0100 NV UP DI PL NZ NA PO NC
堆栈段 答:此时存储器分段的示意图如右图所示。OF、SF、ZF、CF的值都为
31F00
0。
31FF0
2.7 下列操作可使用那些寄存器? 代码段
41FF0 (1) 加法和减法 数据寄存器等
(2) 循环计数 CX 2.6 题的存储器分段示意图 (3) 乘法和除法 AX、DX,乘数和除数用其他寄存
器或存储器
(4) 保存段地址 段寄存器 (5) 表示运算结果为0 ZF=1 (6) 将要执行的指令地址 CS:IP (7) 将要从堆栈取出数据的地址 SS:SP
答:答案见题目的右边。
2.8 那些寄存器可以用来指示存储器地址?
答:BX、BP、SI、DI、堆栈操作时的SP、对应的段地址、386及其后继机型的Exx。
2.9 请将下列左边的项和右边的解释联系起来(把所选字母放在括号中):
(1) CPU (M) A.保存当前栈顶地址的寄存器。 (2) 存储器 (C) B.指示下一条要执行的指令的地址。 (3) 堆栈 (D) C.存储程序、数据等信息的记忆装置,微机有RAM和ROM两种。 (4) IP (B) D.以后进先出方式工作的存储空间。 (5) SP (A) E.把汇编语言程序翻译成机器语言程序的系统程序。 (6) 状态标志 (L) F.唯一代表存储空间中每个字节单元的地址。 (7) 控制标志 (K) G.能被计算机直接识别的语言。 (8) 段寄存器 (J) H.用指令的助记符、符号地址、标号等符号书写程序的语言。 (9) 物理地址 (F) I.把若干个模块连接起来成为可执行文件的系统程序。 (10) 汇编语言 (H) J.保存各逻辑段的起始地址的寄存器,8086/8088机有四个:CS、DS、
SS、ES。
(11) 机器语言 (G) K.控制操作的标志,如DF位。 (12) 汇编程序 (E) L.记录指令操作结果的标志,共6位:OF、SF、ZF、AF、PF、CF。 (13) 连接程序 (I) M.分析、控制并执行指令的部件,由算术逻辑部件ALU和寄存器等
组成。
(14) 指令 (O) N.由汇编程序在汇编过程中执行的指令。 (15) 伪指令 (N) O.告诉CPU要执行的操作(一般还要指出操作数地址),在程序运行时
执行。
答:答案见题目的括号中。
第 三 章. 习 题
3.1 给定(BX)=637DH,(SI)=2A9BH,位移量D=7237H,试确定在以下各种寻址方式下的有效地址是什么?
(1) 立即寻址 (2) 直接寻址
(3) 使用BX的寄存器寻址 (4) 使用BX的简接寻址
(5) 使用BX的寄存器相对寻址 (6) 基址变址寻址 (7) 相对基址变址寻址
答:(1) 操作数在指令中,即立即数;
(2) EA=D=7237H;
(3) 无EA,操作数为(BX)=637DH; (4) EA=(BX)=637DH; (5) EA=(BX)+D=0D5B4H; (6) EA=(BX)+(SI)=8E18H;
(7) EA=(BX)+(SI)+D=1004FH;超过了段的边界,最高进位位丢失,因此EA=004FH。 3.2 试根据以下要求写出相应的汇编语言指令
(1) 把BX寄存器和DX寄存器的内容相加,结果存入DX寄存器中。 (2) 用寄存器BX和SI的基址变址寻址方式把存储器中的一个字节与AL寄存器的内容相加,并把结果送到AL寄存器中。
(3) 用寄存器BX和位移量0B2H的寄存器相对寻址方式把存储器中的一个字和(CX)相加,并把结
果送回存储器中。
(4) 用位移量为0524H的直接寻址方式把存储器中的一个字与数2A59H相加,并把结果送回存储单元中。
(5) 把数0B5H与(AL)相加,并把结果送回AL中。 答:(1) ADD DX, BX
(2) ADD AL, [BX][SI] (3) ADD [BX+0B2H], CX
(4) ADD WORD PTR [0524H], 2A59H (5) ADD AL, 0B5H
3.3 写出把首地址为BLOCK的字数组的第6个字送到DX寄存器的指令。要求使用以下几种寻址方式:
(1) 寄存器间接寻址 (2) 寄存器相对寻址 (3) 基址变址寻址
答:(1) MOV BX, OFFSET BLOCK
ADD BX, (6–1)*2 MOV DX, [BX]
(2) MOV BX, OFFSET BLOCK 改为: MOV BX, (6-1)*2 MOV DX, [BX+(6–1)*2] 也可 MOV DX, BLOCK[BX] (3) MOV BX, OFFSET BLOCK MOV SI, (6–1)*2 MOV DX, [BX][SI] 3.4 现有(DS)=2000H,(BX)=0100H,(SI)=0002H,(20100H)=12H,(20101H)=34H,(20102H)=56H,
(20103H)=78H,(21200H)=2AH,(21201H)=4CH,(21202H)=B7H,(21203H)=65H,试说明下列各条指令执行完后AX寄存器的内容。 (1) MOV AX, 1200H
1B00:2000H 10H (2) MOV AX, BX
1B00:2001H FFH (3) MOV AX, [1200H]
1B00:2002H (4) MOV AX, [BX] 00H 1B00:2003H (5) MOV AX, 1100[BX] 80H (6) MOV AX, [BX][SI] 8000:FF10H (7) MOV AX, 1100[BX][SI] 8000:FF11H ?→(AL) 答:(1) (AX)=1200H ?→(AH)
(2) (AX)=0100H (3) (AX)=4C2AH 3.6题的作图表示 (4) (AX)=3412H (5) (AX)=4C2AH (6) (AX)=7856H (7) (AX)=65B7H 3.5 给定(IP)=2BC0H,(CS)=0200H,位移量D=5119H,(BX)=1200H,(DS)=212AH,(224A0H)=0600H,
(275B9H)=098AH,试为以下的转移指令找出转移的偏移地址。 (1) 段内直接寻址
(2) 使用BX及寄存器间接寻址方式的段内间接寻址 (3) 使用BX及寄存器相对寻址方式的段内间接寻址
答:(1) JMP NEAR PTR 5119H ;(IP)=5119H+((IP)+03H)=7CDCH,物理地址PA=09CDCH
(IP)+03H是JMP NEAR PTR 5119H指令的下一条指令的首地址。
(2) JMP WORD PTR [BX] ;(IP)=((DS)*10H+(BX))=0600H,PA=02600H (3) JMP D[BX] ;(IP)=((DS)*10H+(BX)+D)=098AH,PA=0298AH 3.6 设当前数据段寄存器的内容为1B00H,在数据段的偏移地址2000H单元内,含有一个内容为0FF10H和8000H的指针,它们是一个16位变量的偏移地址和段地址,试写出把该变量装入AX的指令序列,并画图表示出来。 答:MOV BX, [2000H] ;图示如上所示。
MOV BX, 2000H MOV AX, [2000H+2]
LES BX, [BX]
MOV ES, AX
MOV AX, ES :[BX]
MOV AX, ES:[BX]
3.7 在0624H单元内有一条二字节JMP SHORT OBJ指令,如其中位移量为(1) 27H,(2) 6BH,(3) 0C6H,
试问转向地址OBJ的值是多少? 答:(1) OBJ=0624H+02H+27H=064DH
(2) OBJ=0624H+02H+6BH=0691H (3) OBJ=0624H+02H+0C6H=05ECH ;C6H对应的负数为-3AH(向上转移,负位移量) 3.8 假定(DS)=2000H,(ES)=2100H,(SS)=1500H,(SI)=00A0H,(BX)=0100H,(BP)=0010H,数据段中变量名VAL的偏移地址为0050H,试指出下列源操作数字段的寻址方式是什么?其物理地址值是多少?
(1) MOV AX, 0ABH (2) MOV AX, BX (3) MOV AX, [100H] (4) MOV AX, VAL (5) MOV AX, [BX] (6) MOV AX, ES:[BX] (7) MOV AX, [BP] (8) MOV AX, [SI] (9) MOV AX, [BX+10] (10) MOV AX, VAL[BX] (11) MOV AX, [BX][SI] (12) MOV AX, VAL[BX][SI] 答:(1) 立即方式; 操作数在本条指令中
(2) 寄存器寻址方式; 操作数为 (BX)=0100H (3) 直接寻址方式; PA=20100H (4) 直接寻址方式; PA=20050H (5) BX寄存器间接寻址方式; PA=20100H (6) 附加段BX寄存器间接寻址方式; PA=21100H (7) BP寄存器间接寻址方式; PA=15010H (8) SI寄存器间接寻址方式; PA=200A0H (9) BX寄存器相对寻址方式; PA=20110H (10) BX寄存器相对寻址方式; PA=20150H (11) BX和SI寄存器基址变址寻址方式; PA=201A0H (12) BX和SI寄存器相对基址变址寻址方式; PA=201F0H 3.9 在ARRAY数组中依次存储了七个字数据,紧接着是名为ZERO的字单元,表示如下: ARRAY DW 23, 36, 2, 100, 32000, 54, 0 ZERO DW ?
(1) 如果BX包含数组ARRAY的初始地址,请编写指令将数据0传送给ZERO单元。 (2) 如果BX包含数据0在数组中的位移量,请编写指令将数据0传送给ZERO单元。 答:(1) MOV AX, [BX+(7-1)*2]
MOV [BX+(7)*2], AX (2) MOV AX, ARRAY [BX] MOV ARRAY [BX+2], AX
3.10 如TABLE为数据段中0032单元的符号名,其中存放的内容为1234H,试问以下两条指令有什么区
别?指令执行完后AX寄存器的内容是什么? TABLE 0AH MOV AX, TABLE 00H LEA AX, TABLE 14H 答:MOV AX, TABLE是将TABLE单元的内容送到AX,(AX)=1234H TABLE+3 00H
LEA AX,TABLE是将TABLE单元的有效地址送到AX,(AX)=0032H 1EH
00H 3.11 执行下列指令后AX寄存器中的内容是什么?
28H TABLE DW 10, 20, 30, 40, 50 ;000AH, 0014H, 001EH, 0028H,
00H 0032H
32H ENTRY DW 3
┇ 00H MOV BX, OFFSET TABLE 3.11题的TABLEADD BX, ENTRY 存储方式 MOV AX, [BX]
答:(AX)=1E00H (TABLE的存储方式如右图所示)
3.12 下列ASCII码串(包括空格符)依次存储在起始地址为CSTRING的字节单元中:
CSTRING DB ‘BASED ADDRESSING’
请编写指令将字符串中的第1个和第7个字符传送给DX寄存器。 答:MOV DH, CSTRING
MOV DL, CSTRING+7-1
3.13 已知堆栈段寄存器SS的内容是0FFA0H,堆栈指针寄存器SP的内容是00B0H,先执行两条把8057H
和0F79H分别进栈的PUSH指令,再执行一条POP指令。试画出堆栈区和SP的内容变化过程示意图(标出存储单元的物理地址)。
答:堆栈区和SP的内容变化过程示意图如下左图所示。
FFAACH 79H ←再次进栈(SP)←(SP)-2 1B00:2000 8000H
0FH 出栈时(SP)←(SP)+2 1B00:2002 2B00H
FFAAEH 57H ←首次进栈(SP)←(SP)-2 80H FFAB0H 2B00:8000 X
3.14题的存储区情况 3.13题的堆栈区和SP的内容变化过程示意图 3.14 设(DS)=1B00H,(ES)=2B00H,有关存储单元的内容如上右图所示。请写出两条指令把字变量X装
入AX寄存器。
答:MOV BX, [2000H]
MOV AX, ES:[BX] 3.15 求出以下各十六进制数与十六进制数62A0H之和,并根据结果设置标志位SF、ZF、CF和OF的值。
(1) 1234H (2) 4321H (3) CFA0H (4) 9D60H 答:(1) 和为74D4H;SF=0,ZF=0,CF=0,OF=0
(2) 和为A5C1H;SF=1,ZF=0,CF=0,OF=1 (3) 和为3240H;SF=0,ZF=0,CF=1,OF=0 (4) 和为0000H;SF=0,ZF=1,CF=1,OF=0 3.16 求出以下各十六进制数与十六进制数4AE0H的差值,并根据结果设置标志位SF、ZF、CF和OF的
值。
(1) 1234H (2) 5D90H (3) 9090H (4) EA04H 答:(1) 差为C754H;SF=1,ZF=0,CF=1,OF=0
(2) 差为12B0H;SF=0,ZF=0,CF=0,OF=0 (3) 差为45B0H;SF=0,ZF=0,CF=0,OF=1 (4) 差为9F24H;SF=1,ZF=0,CF=0,OF=0 3.17 写出执行以下计算的指令序列,其中X、Y、Z、R、W均为存放16位带符号数单元的地址。
(1) Z←W+(Z-X) (2) Z←W-(X+6)-(R+9) (3) Z←(W*X)/(Y+6),R←余数 (4) Z←((W-X)/5*Y)*2 答:(1) MOV AX, Z ;以下程序都未考虑带符号数的溢出
SUB AX, X ADD AX, W MOV Z, AX (2) MOV BX, X ADD BX, 6 MOV CX, R ADD CR, 9 MOV AX, W SUB AX, BX SUB AX, CX MOV Z, AX (3) ADD Y, 6
MOV AX, W IMUL X IDIV Y MOV Z, AX MOV R, DX (4) MOV AX, W SUB AX, X CWD
MOV BX, 5 IDIV BX IMUL Y
SHL AX, 1 ;((DX),(AX))*2 RCL DX, 1
3.18 已知程序段如下:
MOV AX, 1234H ;(AX)=1234H,标志位不变 MOV CL, 4 ;(AX)和标志位都不变 ROL AX, CL ;(AX)=2341H,CF=1,SF和ZF不变 DEC AX ;(AX)=2340H,CF=1不变,SF=0,ZF=0 MOV CX, 4 ;(AX)和标志位都不变 MUL CX ;(AX)=8D00H,CF=OF=0,其它标志无定义 INT 20H 试问:
(1) 每条指令执行完后,AX寄存器的内容是什么?
(2) 每条指令执行完后,进位、符号和零标志的值是什么? (3) 程序结束时,AX和DX的内容是什么? 答:(1) 见注释;
(2) 见注释;
(3) (AX)=8D00H,(DX)=0
3.19 下列程序段中的每条指令执行完后,AX寄存器及CF、SF、ZF和OF的内容是什么?
MOV AX, 0 ;(AX)=0, 标志位不变 DEC AX ;(AX)=0FFFFH, CF不变,SF=1,ZF=0,OF=0 ADD AX, 7FFFH ;(AX)=7FFEH, CF=1,SF=0,ZF=0,OF=0 ADD AX, 2 ;(AX)=8000H, CF=0,SF=1,ZF=0,OF=1 NOT AX ;(AX)=7FFFH, 标志位不变 SUB AX, 0FFFFH ;(AX)=8000H, CF=1,SF=1,ZF=0,OF=1 ADD AX, 8000H ;(AX)=0, CF=1,SF=0,ZF=1,OF=1 SUB AX, 1 ;(AX)=0FFFFH, CF=1,SF=1,ZF=0,OF=0 AND AX, 58D1H ;(AX)=58D1H, CF=0,SF=0,ZF=0,OF=0 SAL AX, 1 ;(AX)=0B1A2H, CF=0,SF=1,ZF=0,OF=1 SAR AX, 1 ;(AX)=0D8D1H, CF=0,SF=1,ZF=0,OF=0 NEG AX ;(AX)= 272FH, CF=1,SF=0,ZF=0,OF=0 ROR AX, 1 ;(AX)= 9397H, CF=1,SF和ZF不变,OF=1 答:见注释。 3.20 变量DATAX和变量DATAY的定义如下:
DATAX DW 0148H
DW 2316H
DATAY DW 0237H
DW 4052H
请按下列要求写出指令序列:
(1) DATAX和DATAY两个字数据相加,和存放在DATAY中。
(2) DATAX和DATAY两个双字数据相加,和存放在从DATAY开始的双字单元中。 (3) 解释下列指令的作用:
STC
MOV BX, DATAX ADC BX, DATAY
(4) DATAX和DATAY两个字数据相乘(用MUL)。 (5) DATAX和DATAY两个双字数据相乘(用MUL)。 (6) DATAX除以23(用DIV)。
(7) DATAX双字除以字DATAY (用DIV)。 答:(1) MOV AX, DATAX
ADD DATAY, AX MOV AX, DATAX+2 ADD DATAY+2, AX (2) MOV AX, DATAX
ADD DATAY, AX MOV AX, DATAX+2 ADC DATAY+2, AX MOV DATAY+4, 0 ;用于存放进位位 ADC DATAY+4, 0
(3) DATAX和DATAY两个字数据之和加1,结果存入BX寄存器。 (4) RESULT1 DW 0
DW 0
RESULT2 DW 0
DW 0 ┇
MOV AX, DATAX MUL DATAY
MOV RESULT1 , AX MOV RESULT1+2, DX MOV AX, DATAX+2 MUL DATAY+2
MOV RESULT2 , AX MOV RESULT2+2, DX (5) AA DW 0
BB DW 0 CC DW 0 DD DW 0
┇
MOV AX, DATAX MUL DATAY MOV AA , AX MOV BB, DX MOV AX, DATAX MUL DATAY+2 ADD BB, AX ADC CC, DX MOV AX, DATAX+2 MUL DATAY ADD BB, AX ADC CC, DX ADC DD, 0
MOV AX, DATAX+2 MUL DATAY+2 ADD CC, AX ADC DD, DX
(6) MOV AX, DATAX
MOV BL, 23 DIV BL
(7) MOV DX, DATAX+2
MOV AX, DATAX DIV DATAY
3.21 写出对存放在DX和AX中的双字长数求补的指令序列。
答:NEG DX 也可为: NOT DX
NEG AX NOT AX SBB DX, 0 ADD AX, 1
ADC DX, 0
3.22 试编写一程序求出双字长数的绝对值。双字长数在A和A+2单元中,结果存放在B和B+2单元中。
答:程序段如下:
MOV AX, A MOV DX, A+2 CMP DX, 0 JNS ZHENSHU ;不是负数则转走 NEG DX NEG AX SBB DX, 0
ZHENSHU: MOV B, AX
MOV B+2, DX INT 20H 3.23 假设(BX)=0E3H,变量VALUE中存放的内容为79H,确定下列各条指令单独执行后的结果。
(1) XOR BX, VALUE ;(BX)=9AH,CF、OF都为0,AF无定义,SF=1,ZF=0,PF=1 (2) AND BX, VALUE ;(BX)=61H,CF、OF都为0,AF无定义,SF=0,ZF=0,PF=0 (3) OR BX, VALUE ;(BX)=0FBH,CF、OF都为0,AF无定义,SF=1,ZF=0,PF=0 (4) XOR BX, 0FFH ;(BX)=1CH,CF、OF都为0,AF无定义,SF=0,ZF=0,PF=0 (5) AND BX, 0 ;(BX)=00H,CF、OF都为0,AF无定义,SF=0,ZF=1,PF=1 (6) TEST BX, 01H ;(BX)=0E3H,CF、OF都为0,AF无定义,SF=1,ZF=0,PF=0 答:见注释。 3.24 试写出执行下列指令序列后BX寄存器的内容。执行前(BX)=6D16H。
MOV CL, 7 SHR BX, CL
答:(BX)=00DAH。
3.25 试用移位指令把十进制数+53和-49分别乘以2。它们应该用什么指令?得到的结果是什么?如果
要除以2呢? 答:MOV AL, 53
SAL AL, 1 ;(AL)=(+53*2)=6AH MOV AL, -49 SAL AL, 1 ;(AL)=(-49*2)=9EH MOV AL, 53 SAR AL, 1 ;(AL)=(53/2)= 1AH MOV AL, -49 SAR AL, 1 ;(AL)=(-49/2)=0E7H 3.26 试分析下面的程序段完成什么功能? MOV CL, 04 SHL DX, CL MOV BL, AH
SHL AX, CL SHR BL, CL OR DL, BL
答:本程序段将 ((DX),(AX)) 的双字同时左移4位,即将此双字乘以10H (16)。 3.27 假定(DX)=0B9H,(CL)=3,(CF)=1,确定下列各条指令单独执行后DX中的值。
(1) SHR DX, 1 ;(DX)=05CH (2) SAR DX, CL ;(DX)=17H (3) SHL DX, CL ;(DX)=5C8H (4) SHL DL, 1 ;(DX)=72H (5) ROR DX, CL ;(DX)=2017H (6) ROL DL, CL ;(DX)=0CDH (7) SAL DH, 1 ;(DX)=0B9H (8) RCL DX, CL ;(DX)=2CCH (4) RCR DL, 1 ;(DX)=0DCH 答:见注释。 3.28 下列程序段执行完后,BX寄存器的内容是什么?
MOV CL, 3 MOV BX, 0B7H ROL BX,1 ROR BX, CL
答:(BX)=0C02DH。
3.29 假设数据段定义如下:
CONAME DB ‘SPACE EXPLORERS INC.’ PRLINE DB 20 DUP (‘’)
用串指令编写程序段分别完成以下功能:
(1) 从左到右把CONAME中的字符串传送到PRLINE。 (2) 从右到左把CONAME中的字符串传送到PRLINE。 (3) 把CONAME中的第3和第4个字节装入AX。
(4) 把AX寄存器的内容存入从PRLINE+5开始的字节中。
(5) 检查CONAME字符串中有无空格字符,如有则把第一个空格字符的地址传送给BX寄存器。 答:(1) MOV CX, 20
CLD MOV SI, SEG CONAME MOV DS, SI MOV ES, SI LEA SI, CONAME LEA DI, PRLINE REP MOVSB (2) MOV CX, 20 STD MOV SI, SEG CONAME MOV DS, SI MOV ES, SI LEA SI, CONAME ADD SI, 20-1 LEA DI, PRLINE ADD DI, 20-1 REP MOVSB (3) MOV AX, WORD PTR CONAME+3-1 (4) MOV WORD PTR PRLINE +5, AX (5) MOV AL, ‘ ’ ;空格的ASCII码送AL寄存器 CLD
MOV DI, SEG CONAME MOV ES, DI LEA DI, CONAME REPNE SCASB JNE NEXT DEC DI MOV BX, DI NEXT: ┇
3.30 编写程序段,把字符串STRING中的‘&’字符用空格符代替。 STRING DB ‘The date is FEB&03’ 答:程序段如下:
MOV CX, 18 MOV AL, ‘&’ CLD
MOV DI, SEG STRING MOV ES, DI LEA DI, STRING REPNE SCASB JNE NEXT DEC DI
MOV ES:BYTE PTR [DI], ‘ ’ ;送空格符 NEXT: ┇
3.31 假设数据段中数据定义如下:
STUDENT_NAME DB 30 DUP (?) STUDENT_ADDR DB 9 DUP (?) PRINT_LINE DB 132 DUP (?) 分别编写下列程序段:
(1) 用空格符清除PRINT_LINE域。
(2) 在STUDENT_ADDR中查找第一个‘-’。 (3) 在STUDENT_ADDR中查找最后一个‘-’。
(4) 如果STUDENT_NAME域中全是空格符时,填入‘*’。
(5) 把STUDENT_NAME移到PRINT_LINE的前30个字节中,把STUDENT_ ADDR移到PRINT_LINE的
后9个字节中。
答:公共的程序段如下:
MOV DI, DS MOV ES, DI (1) MOV CX, 132
MOV AL., ‘ ’ ;空格的ASCII码送AL寄存器 CLD LEA DI, PRINT_LINE REP STOSB (2) MOV CX, 9
MOV AL., ‘-’ CLD LEA DI, STUDENT_ ADDR REPNE SCASB JNE NO_DASH DEC DI NO_DASH: ┇ (3) MOV CX, 9 MOV AL., ‘-’ STD
LEA DI, STUDENT_ ADDR ADD DI, 9-1 REPNE SCASB JNE NO_DASH INC DI NO_DASH: ┇ (4) MOV CX, 30 MOV AL, ‘ ’ CLD LEA DI, STUDENT_NAME REPE SCASB JNE NEXT MOV CX, 30 MOV AL, ‘*’ LEA DI, STUDENT_NAME REP STOSB NEXT: ┇ (5) MOV CX, 30 CLD LEA SI, STUDENT_NAME LEA DI, PRINT_LINE REP MOVSB MOV CX, 9 STD LEA SI, STUDENT_ADDR+9-1 LEA DI, PRINT_LINE+132-1 REP MOVSB
;空格的ASCII码送AL寄存器
;“*”的ASCII码送AL寄存器
3.32 编写一程序段:比较两个5字节的字符串OLDS和NEWS,如果OLDS字符串不同于NEWS字符串则执
行NEW_LESS;否则顺序执行程序。 答:程序段如下:
MOV CX, 5 CLD
MOV DI, SEG OLDS MOV DS, DI MOV ES, DI LEA SI, OLDS LEA DI, NEWS REPE CMPSB JNE NEW_LESS
┇
NEW_LESS: ┇ 3.33 假定AX和BX中的内容为带符号数,CX和DX中的内容为无符号数,请用比较指令和条件转移指令
实现以下判断:
(1) 若DX的内容超过CX的内容,则转去执行EXCEED。 (2) 若BX的内容大于AX的内容,则转去执行EXCEED。 (3) 若CX的内容等于0,则转去执行ZERO。
(4) BX与AX的内容相比较是否产生溢出?若溢出则转OVERFLOW。 (5) 若BX的内容小于等于AX的内容,则转EQ_SMA。 (6) 若DX的内容低于等于CX的内容,则转EQ_SMA。 答:(1) CMP DX, CX
JA EXCEED (2) CMP BX, AX
JG EXCEED (3) JCXZ ZERO (4) CMP BX, AX
JO OVERFLOW (5) CMP BX, AX
JLE EQ_SMA (6) CMP DX, CX
JBE EQ_SMA
3.34 试分析下列程序段:
ADD AX, BX JNO L1 JNC L2 SUB AX, BX JNC L3 JNO L4 JMP SHORT L5
如果AX和BX的内容给定如下:
AX BX
(1) 147BH 80DCH (2) B568H 42C8H (3) 42C8H 608DH (4) D023H 9FD0H (5) 94B7H B568H
问该程序分别在上面5种情况下执行后,程序转向哪里? 答:(1) 转向L1
(2) 转向L1 (3) 转向L2 (4) 转向L5 ;因为加法指令后AX中已经是6FF3H (5) 转向L5 ;因为加法指令后AX中已经是4A14H
3.35 指令CMP AX, BX后面跟着一条格式为J… L1的条件转移指令,其中…可以是B、NB、BE、NBE、L、NL、LE、NLE中的任意一个。如果AX和BX的内容给定如下:
AX BX (1) 1F52H 1F52H (2) 88C9H 88C9H (3) FF82H 007EH (4) 58BAH 020EH (5) FFC5H FF8BH (6) 09A0H 1E97H (7) 8AEAH FC29H (8) D367H 32A6H
问以上8条转移指令中的哪几条将引起转移到L1? 答:(1) JNB、JBE、JNL、JLE
(2) JNB、JBE、JNL、JLE (3) JNB、JNBE、JL、JLE (4) JNB、JNBE、JNL、JNLE (5) JNB、JNBE、JL、JLE (6) JB、JBE、JL、JLE (7) JB、JBE、JNL、JNLE (8) JNB、JNBE、JL、JLE 3.36 假设X和X+2单元的内容为双精度数p,Y和Y+2单元的内容为双精度数q,(X和Y为低位字)试
说明下列程序段做什么工作?
MOV DX, X+2
MOV AX, X ADD AX, X ADC DX, X+2 CMP DX, Y+2 JL L2 JG L1 CMP AX, Y JBE L2
L1: MOV AX, 1
JMP SHORT EXIT
L2: MOV AX, 2 EXIT:INT 20H
答:此程序段判断p*2>q,则使(AX)=1后退出;p*2≤q,则使(AX)=2后退出。
3.37 要求测试在STATUS中的一个字节,如果第1、3、5位均为1则转移到ROUTINE_1;如果此三位中
有两位为1则转移到ROUTINE_2;如果此三位中只有一位为1则转移到ROUTINE_3;如果此三位全为0则转移到ROUTINE_4。试画出流程图,并编制相应的程序段。 答:程序段如下:
程序开始 MOV AL, STATUS AND AL, 00010101B ;只保留第1、3、5位 (AL)←(STATUS)只保留第1,3,5JZ ROUTINE_4 ;3位全为0转ROUTINE_4 JPE ROUTINE_2 ;两位为1转ROUTINE_2 全为0吗? CMP AL, 00010101B
Y JZ ROUTINE_1 ;3位全为1转ROUTINE_1 N ROUTINE_3: ┇ ;仅一位为1执行ROUTINE_3 转去执行二位为1吗? JMP EXIT
ROUTINE_4 Y ROUTINE_1: ┇
N JMP EXIT
Y ROUTINE_2: ┇ 全为1吗? JMP EXIT N 转去执行转去执行ROUTINE_4: ┇
ROUTINE_2 仅一位为1执行ROUTINE_1 EXIT: INT 20H
ROUTINE_3 3.38 在下列程序的括号中分别填入如下指令:
(1) LOOP L20 (2) LOOPE L20
执行EXIT程序结束 (3) LOOPNE L20
试说明在三种情况下,当程序执行完后,AX、BX、 3.44题的程序流程图
CX、DX四个寄存器的内容分别是什么? TITLE EXLOOP.COM CODESG SEGMENT
ASSUME CS:CODESG, DS: CODSEG, SS: CODSEG ORG 100H
BEGIN: MOV AX, 01
MOV BX, 02 MOV DX, 03 MOV CX, 04
L20:
INC AX ADD BX, AX SHR DX, 1 ( ) RET
CODESG ENDS
END BEGIN
答:(1) (AX)=5H,(BX)=10H,(CX)=0H,(DX)=0H
(2) (AX)=2H,(BX)=4H,(CX)=3H,(DX)=1H (3) (AX)=3H,(BX)=7H,(CX)=2H,(DX)=0H
3.39 考虑以下的调用序列:
(1) MAIN调用NEAR的SUBA过程(返回的偏移地址为0400); (2) SUBA调用NEAR的SUBB过程(返回的偏移地址为0A00);
(3) SUBB调用FAR的SUBC过程(返回的段地址为B200,返回的偏移地址为0100); (4) 从SUBC返回SUBB;
(5) SUBB调用NEAR的SUBD过程(返回的偏移地址为0C00); (6) 从SUBD返回SUBB; (7) 从SUBB返回SUBA; (8) 从SUBA返回MAIN;
(9) 从MAIN调用SUBC(返回的段地址为1000,返回的偏移地址为0600); 请画出每次调用及返回时的堆栈状态。
答:每次调用及返回时的堆栈状态图如下所示:
00 新SP位置 00 原SP位置 01 01 00 00 00 新SP位置 B2 B2 0A 00 新SP位置 00 原SP位置 00 新SP位置 00 原SP位置 04 0A 0A 04 原SP位置 00 00 (2) SUBA调 (1) MAIN调 (3) SUBB调 (4) 从SUBC 用SUBB过程 SUBA过程 用用SUBC过程 返回SUBB
00 00 00 00 00 01 01 01 01 01 00 新SP位置 00 原SP位置 00 00 00 0C 0C 0C 0C 0C 00 原SP位置 00 新SP位置 00 原SP位置 00 00 新SP位置 0A 0A 0A 0A 06 00 00 00 新SP位置 00 原SP位置 00 04 04 04 04 10 新SP位置 原SP位置
(5) SUBB调 (6) 从SUBD (7) 从SUBB (8) 从SUBA (9) 从MAIN
用SUBD过程 返回SUBB 返回SUBA 返回MAIN 调用SUBC
3.40 假设(EAX)=00001000H,(EBX)=00002000H,(DS)=0010H,试问下列指令访问内存的物理地址是什
么?
(1) MOV ECX,[EAX+EBX] (2) MOV [EAX+2*EBX],CL
(3) MOV DH,[EBX+4*EAX+1000H]
答:(1) PA=(DS)*10H+EA=00100H+00001000H+00002000H=00003100H
(2) PA=(DS)*10H+EA=00100H+00001000H+2*00002000H=00005100H
(3) PA=(DS)*10H+EA=00100H+00002000H+4*00001000H+1000H=00007100H 3.41 假设(EAX)=9823F456H,(ECX)=1F23491H,(BX)=348CH,(SI)=2000H,(DI)=4044H。在DS段中从
偏移地址4044H单元开始的4个字节单元中,依次存放的内容为92H,6DH,0A2H和4CH,试问下列各条指令执行完后的目的地址及其中的内容是什么? (1) MOV [SI],EAX (2) MOV [BX],ECX (3) MOV EBX,[DI]
答:(1) 目的地址为DS:2000H,内容依次为:56H,0F4H,23H和98H
(2) 目的地址为DS:348CH,内容依次为:91H,34H,0F2H和01H (3) 目的操作数为EBX寄存器,(EBX)=4CA26D92H
3.42 说明下列指令的操作
(1) PUSH AX (2) POP ESI (3) PUSH [BX] (4) PUSHAD (5) POP DS (6) PUSH 4 答:见注释。
;将(AX)压入堆栈 ;将堆栈中的双字弹出到ESI寄存器中 ;将((BX))对应存储单元中的字压入堆栈 ;32位通用寄存器依次进栈 ;将堆栈中的字弹出到DS寄存器中 ;将立即数4以字的方式压入堆栈
3.43 请给出下列各指令序列执行完后目的寄存器的内容。
(1) MOV EAX,299FF94H
ADD EAX,34FFFFH ;(EAX)= 2CEFF93H (2) MOV EBX,40000000
SUB EBX,1500000 ;(EBX)= 3EB00000H (3) MOV EAX,39393834H
AND EAX,0F0F0F0FH ;(EAX)= 09090804H (4) MOV EDX,9FE35DH
XOR EDX,0F0F0F0H ;(EDX)= 6F13ADH 答:见注释。 3.44 请给出下列各指令序列执行完后目的寄存器的内容。
(1) MOV BX,-12
MOVSX EBX,BX ;(EBX)= 0FFFF FFF4H (2) MOV CL,-8
MOVSX EDX,CL ;(EDX)= 0FFFF FFF8H (3) MOV AH,7
MOVZX ECX,AH ;(ECX)= 0000 0007H (4) MOV AX,99H
MOVZX EBX,AX ;(EBX)= 0000 0099H 答:见注释。 3.45 请给出下列指令序列执行完后EAX和EBX的内容。
MOV ECX,307 F455H BSF EAX,ECX ;(EAX)= 0D BSR EBX,ECX ;(EBX)= 25D 答:见注释。 3.46 请给出下列指令序列执行完后AX和DX的内容。
MOV BX,98H BSF AX,BX ;(AX)= 3D BSR DX,BX ;(DX)= 7D 答:见注释。
3.47 请编写一程序段,要求把ECX、EDX和ESI的内容相加,其和存入EDI寄存器中(不考虑溢出)。
答:MOV EDI,0 也可为: MOV EDI,ECX
ADD EDI,ECX ADD EDI,EDX ADD EDI,EDX ADD EDI,ESI ADD EDI,ESI 3.48 请说明IMUL BX,DX,100H指令的操作。
答:(BX)←(DX)*100H
3.49 试编写一程序段,要求把BL中的数除以CL中的数,并把其商乘以2,最后的结果存入DX寄存器
中。
答:MOV AL,BL
MOV AH,0 ;假定为无符号数,否则用CBW指令即可 DIV CL
MOV AH,0 SHL AX,1 MOV DX,AX
3.50 请说明JMP DI和JMP [DI]指令的区别。
答:JMP DI是转移到以(DI)内容为偏移地址的单元去执行指令;JMP [DI]是转移到以(DI)间接
寻址的内存单元内容为偏移地址的单元去执行指令。 3.51 试编写一程序段,要求在长度为100H字节的数组中,找出大于42H的无符号数的个数并存入字节
单元UP中;找出小于42H的无符号数的个数并存入字节单元DOWN中。 答:JMP BEGIN
UP DB 0
存储器 DOWNDB 0
002EH 新(SP) TABLE DB 100H DUP (?) ;数组
002FH BEGIN:
0030H MOV CX,100H
0031H MOV BX,-1
0032H MOV SI,0 0033H MOV DI,0 0034H 堆 L1:INC BX 0035H 栈 0036H CMP TABLE[BX],42H
帧
0037H JA L2
0038H JB L3
0039H JMP L4
003AH L2:INC SI 003BH JMP L4 003CH L3:INC DI 003DH L4:LOOP L1 新(BP) 原(BP)低字节 003EH MOV UP,SI 原(BP)高字节 003FH 0040H 原(SP) MOV DOWN,DI 3.52 请用图表示ENTER 16,0所生成的堆栈帧的情况。
答:答案见右图。
3.52题的答案 第 四 章. 习 题
4.1 指出下列指令的错误:
(1) MOV AH, BX (2) MOV [BX], [SI] (3) MOV AX, [SI][DI] (4) MOV MYDAT [BX][SI], ES:AX (5) MOV BYTE PTR [BX], 1000 (6) MOV BX, OFFSET MYDAT [SI] (7) MOV CS, AX (8) MOV ECX, AX 答:见注释。
;寄存器类型不匹配 ;不能都是存储器操作数 ;[SI]和[DI]不能一起使用 ;AX寄存器不能使用段超越 ;1000超过了一个字节的范围
;MYDAT [SI]已经是偏移地址,不能再使用OFFSET ;CS不能用作目的寄存器 ;两个操作数的数据类型不同
4.2 下面哪些指令是非法的?(假设OP1,OP2是已经用DB定义的变量)
(1) CMP 15, BX ;错,立即数不能作为目的操作数 (2) CMP OP1, 25 (3) CMP OP1, OP2 ;错,不能都是存储器操作数 (4) CMP AX, OP1 ;错,类型不匹配,应为CMP ax, word ptr op1 答:见注释。
4.3 假设下列指令中的所有标识符均为类型属性为字的变量,请指出下列哪些指令是非法的?它们的
错误是什么?
(1) MOV BP, AL ;错,寄存器类型不匹配 (2) MOV WORD_OP [BX+4*3][DI], SP (3) MOV WORD_OP1, WORD_OP2 ;错,不能都是存储器操作数 (4) MOV AX, WORD_OP1[DX] ;错,DX不能用于存储器寻址 (5) MOV SAVE_WORD, DS
(6) MOV SP, SS:DATA_WORD [BX][SI] (7) MOV [BX][SI], 2 ;错,[BX][SI]未指出数据类型 (8) MOV AX, WORD_OP1+WORD_OP2 (9) MOV AX, WORD_OP1-WORD_OP2+100 (10) MOV WORD_OP1, WORD_OP1-WORD_OP2 答:见注释。 4.4 假设VAR1和VAR2为字变量,LAB为标号,试指出下列指令的错误之处:
(1) ADD VAR1, VAR2 ;不能都是存储器操作数 (2) SUB AL, VAR1 ;数据类型不匹配 (3) JMP LAB [SI] ;LAB是标号而不是变量名,后面不能加[SI] (4) JNZ VAR1 ;VAR1是变量而不是标号 (5) JMP NEAR LAB ;应使用NEAR PTR 答:见注释。 4.5 画图说明下列语句所分配的存储空间及初始化的数据值。
(1) BYTE_VAR DB ‘BYTE’,12,-12H,3 DUP(0,?,2 DUP(1,2),?) (2) WORD_VAR DW 5 DUP(0,1,2),?,-5,‘BY’,‘TE’,256H 答:答案如下图所示。
4.6 试列出各种方法,使汇编程序把5150H存入一个存储器字中(如:DW 5150H)。
答:DW 5150H
BYTE_VAR 42H WORD_VAR
59H DB 50H, 51H
54H
45H DB ‘PQ’
0DH
EEH DW ‘QP’
00H 将上面
- 内容再 ORG 5150H 01H 重复4次 DW $ 02H
01H 4.7 请设置一个数据段DATASG,其中定义以下字符变量或数据变量。
02H
(1) FLD1B为字符串变量:‘personal computer’;
-
(2) FLD2B为十进制数字节变量:32; 00H (3) FLD3B为十六进制数字节变量:20; - (4) FLD4B为二进制数字节变量:01011001; 01H 02H (5) FLD5B为数字的ASCII字符字节变量:32654;
01H (6) FLD6B为10个零的字节变量;
02H (7) FLD7B为零件名(ASCII码)及其数量(十进制数)的表格:
- PART1 20 PART2 50 4.5题答案 PART3 14
(8) FLD1W为十六进制数字变量:FFF0; (9) FLD2W为二进制数的字变量:01011001; (10) FLD3W为(7)零件表的地址变量;
(11) FLD4W为包括5个十进制数的字变量:5,6,7,8,9; (12) FLD5W为5个零的字变量;
(13) FLD6W为本段中字数据变量和字节数据变量之间的地址差。
00H 00H 01H 00H 02H 00H ┇ ┇ ┇ - - FBH FFH 00H 59H 42H 45H 54H 56H 02H
答:DATASG SEGMENT
FLD1B FLD2B FLD3B FLD4B FLD5B FLD6B FLD7B
DB ‘personal computer’ DB 32 DB 20H
DB 01011001B DB ‘32654’ DB 10 DUP (0) DB ‘PART1’, 20 DB ‘PART2’, 50 DB ‘PART3’, 14
FLD1W DW 0FFF0H FLD2W DW 01011001B FLD3W DW FLD7B
FLD4W DW 5, 6, 7, 8, 9 FLD5W DW 5 DUP (0) FLD6W DW FLD1W-FLD1B
DATASG ENDS
4.8 假设程序中的数据定义如下:
PARTNO DW ? PNAME DB 16 DUP (?) COUNT DD ? PLENTH EQU $-PARTNO
问PLENTH的值为多少?它表示什么意义?
答:PLENTH=22=16H,它表示变量PARTNO、PNAME、COUNT总共占用的存储单元数(字节数)。 4.9 有符号定义语句如下:
BUFF DB 1, 2, 3, ‘123’ EBUFF DB 0 L EQU EBUFF - BUFF 问L的值是多少? 答:L=6。
4.10 假设程序中的数据定义如下:
LNAME DB 30 DUP (?) ADDRESS DB 30 DUP (?) CITY DB 15 DUP (?) CODE_LIST DB 1, 7, 8, 3, 2
(1) 用一条MOV指令将LNAME的偏移地址放入AX。
(2) 用一条指令将CODE_LIST的头两个字节的内容放入SI。
(3) 用一条伪操作使CODE_LENGTH的值等于CODE_LIST域的实际长度。 答:(1) MOV AX, OFFSET LNAME
(2) MOV SI, WORD PTR CODE_LIST
(3) CODE_LENGTH EQU $ - CODE_LIST ;此语句必须放在CODE_LIST语句之后
4.11 试写出一个完整的数据段DATA_SEG,它把整数5赋予一个字节,并把整数-1,0,2,5和4放在
10字数组DATA_LIST的头5个单元中。然后,写出完整的代码段,其功能为:把DATA_LIST中头5个数中的最大值和最小值分别存入MAX和MIN单元中。 答:DATA_SEG SEGMENT
NUM DB 5
DATA_LIST DW -1, 0, 2, 5, 4, 5 DUP (?) MAX DW ? MIN DW ? DATA_SEG ENDS
;----------------------------------------------------------------
CODE_SEG SEGMENT MAIN PROC FAR
ASSUME CS: CODE_SEG, DS: DATA_SEG
START: PUSH DS ;设置返回DOS
SUB AX, AX PUSH AX
MOV AX, DATA_SEG ;给DS赋值 MOV DS, AX
;
MOV CX, 4 ;程序段开始 LEA BX, DATA_LIST MOV AX, [BX] MOV MAX, AX MOV MIN, AX
ROUT1: ADD BX, 2
MOV AX, [BX] CMP AX, MAX JNGE ROUT2 MOV MAX, AX
ROUT2: CMP AX, MIN
JNLE ROUT3 MOV MIN, AX
ROUT3: LOOP ROUT1 ;程序段结束
RET
MAIN ENDP CODE_SEG ENDS
;----------------------------------------------------------------
END START
4.12 给出等值语句如下:
ALPHA EQU 100 BETA EQU 25 GAMMA EQU 2 下列表达式的值是多少? (1) ALPHA * 100 + BETA (2) ALPHA MOD GAMMA + BETA (3) (ALPHA +2) * BETA – 2 (4) (BETA / 3) MOD 5 (5) (ALPHA +3) * (BETA MOD GAMMA) (6) ALPHA GE GAMMA (7) BETA AND 7 (8) GAMMA OR 3 答:见注释。
;=2729H ;=19H ;=9F4H ;=3H ;=67H ;=0FFFFH ;=01H ;=03H
4.13 对于下面的数据定义,三条MOV指令分别汇编成什么?(可用立即数方式表示)
TABLEA DW 10 DUP (?) TABLEB DB 10 DUP (?) TABLEC DB ‘1234’ ┇ MOV AX, LENGTH TABLEA ;汇编成MOV AX, 000AH MOV BL, LENGTH TABLEB ;汇编成MOV BL, 000AH MOV CL, LENGTH TABLEC ;汇编成MOV CL, 0001H 答:见注释。 4.14 对于下面的数据定义,各条MOV指令单独执行后,有关寄存器的内容是什么?
FLDB DB ?
TABLEA DW 20 DUP (?) TABLEB DB ‘ABCD’ (1) MOV AX, TYPE FLDB (2) MOV AX, TYPE TABLEA (3) MOV CX, LENGTH TABLEA (4) MOV DX, SIZE TABLEA (5) MOV CX, LENGTH TABLEB 答:见注释。
;(AX)=0001H ;(AX)=0002H ;(CX)=0014H ;(DX)=0028H ;(CX)=0001H
4.15 指出下列伪操作表达方式的错误,并改正之。
(1) DATA_SEG SEG ;DATA_SEG SEGMENT(伪操作错) (2) SEGMENT ‘CODE’ ;SEGNAME SEGMENT ‘CODE’(缺少段名字) (3) MYDATA SEGMENT/DATA ;MYDATA SEGMENT
┇ ENDS ;MYDATA ENDS(缺少段名字)
(4) MAIN_PROC PROC FAR ;删除END MAIN_PROC也可以
┇
END MAIN_PROC ;MAIN_PROC ENDP ;上下两句交换位置
MAIN_PROC ENDP ; END MAIN_PROC 答:见注释。 4.16 按下面的要求写出程序的框架
(1) 数据段的位置从0E000H开始,数据段中定义一个100字节的数组,其类型属性既是字又是字
节;
(2) 堆栈段从小段开始,段组名为STACK;
(3) 代码段中指定段寄存器,指定主程序从1000H开始,给有关段寄存器赋值; (4) 程序结束。
答:程序的框架如下:
DATA_SEG SEGMENT AT 0E000H ARRAY_B LABEL BYTE ARRAY_W DW 50 DUP (?) DATA_SEG ENDS ;以上定义数据段
;---------------------------------------------------------------- STACK_SEG SEGMENT PARA STACK ‘STACK’
DW 100H DUP (?)
TOS LABEL WORD STACK_SEG ENDS ;以上定义堆栈段
;---------------------------------------------------------------- CODE_SEG SEGMENT MAIN PROC FAR
ASSUME CS: CODE_SEG, DS: DATA_SEG, SS: STACK_SEG ORG 1000H
START: MOV AX, STACK_SEG
MOV SS, AX ;给SS赋值 MOV SP, OFFSET TOS ;给SP赋值 PUSH DS ;设置返回DOS SUB AX, AX PUSH AX
MOV AX, DATA_SEG MOV DS, AX ;给DS赋值 ┇ ;程序段部分 RET
MAIN ENDP
CODE_SEG ENDS ;以上定义代码段
;----------------------------------------------------------------
END START
4.17 写一个完整的程序放在代码段C_SEG中,要求把数据段D_SEG中的AUGEND和附加段E_SEG中的
ADDEND相加,并把结果存放在D_SEG 段中的SUM中。其中AUGEND、ADDEND和SUM均为双精度数,AUGEND赋值为99251,ADDEND赋值为 -15962。 答:程序如下:
D_SEG SEGMENT AUGW LABEL WORD AUGEND DD 99251 SUM DD ? D_SEG ENDS ;以上定义数据段
;---------------------------------------------------------------- E_SEG SEGMENT ADDW LABEL WORD ADDEND DD -15962 E_SEG ENDS ;以上定义附加段
;---------------------------------------------------------------- C_SEG SEGMENT MAIN PROC FAR
ASSUME CS: C_SEG, DS: D_SEG, ES: E_SEG
START: PUSH DS ;设置返回DOS
SUB AX, AX PUSH AX
MOV AX, D_SEG MOV DS, AX ;给DS赋值 MOV AX, E_SEG MOV ES, AX ;给ES赋值
;
MOV AX, AUGW ;以下6条指令进行加法计算 MOV BX, AUGW+2 ADD AX, ES: ADDW
ADC BX, ES: ADDW+2 ;不考虑有符号数溢出 MOV WORD PTR SUM, AX MOV WORD PTR [SUM+2], BX RET
MAIN ENDP C_SEG ENDS ;以上定义代码段
;----------------------------------------------------------------
END START 4.18 请说明表示程序结束的微操作和结束程序执行的语句之间的差别。它们在源程序中应如何表示?
答:表示程序结束的微操作是指示汇编程序MASM结束汇编的标志,在源程序中用END表示;结束
程序执行的语句是结束程序运行而返回操作系统的指令,在源程序中有多种表示方法,比如INT 20H或MOV AX, 4C00H INT 21H以及RET等。 4.19 试说明下述指令中哪些需要加上PTR操作符:
BVAL DB 10H,20H WVAL DW 1000H (1) MOV AL,BVAL ;不需要 (2) MOV DL,[BX] ;不需要 (3) SUB [BX],2 ;需要,如SUB BYTE PTR [BX],2 (4) MOV CL,WVAL ;需要,如MOV CL,BYTE PTR WVAL (5) ADD AL,BVAL+1 ;不需要
答:见注释。
第 五 章. 习 题
5.1 试编写一个汇编语言程序,要求对键盘输入的小写字母用大写字母显示出来。
答:程序段如下:
BEGIN: MOV AH, 1 ;从键盘输入一个字符的DOS调用
INT 21H
CMP AL, ‘a’ ;输入字符<‘a’吗? JB STOP
CMP AL, ‘z’ ;输入字符>‘z’吗? JA STOP SUB AL, 20H ;转换为大写字母,用AND AL, 1101 1111B也可 MOV DL, AL ;显示一个字符的DOS调用 MOV AH, 2 INT 21H JMP BEGIN
STOP: RET 5.2 编写程序,从键盘接收一个小写字母,然后找出它的前导字符和后续字符,再按顺序显示这三个字符。
答:程序段如下:
BEGIN: MOV AH, 1 ;从键盘输入一个字符的DOS调用
INT 21H
CMP AL, ‘a’ ;输入字符<‘a’吗? JB STOP
CMP AL, ‘z’ ;输入字符>‘z’吗? JA STOP DEC AL ;得到前导字符 MOV DL, AL ;准备显示三个字符 MOV CX, 3
DISPLAY: MOV AH, 2 ;显示一个字符的DOS调用
INT 21H INC DL LOOP DISPLAY
STOP: RET 5.3 将AX寄存器中的16位数分成4组,每组4位,然后把这四组数分别放在AL、BL、CL和DL中。 答:程序段如下:
DSEG SEGMENT STORE DB 4 DUP (?) DSEG ENDS
┇
BEGIN: MOV CL, 4 ;右移四次
MOV CH, 4 ;循环四次 LEA BX, STORE
A10: MOV DX, AX
AND DX, 0FH ;取AX的低四位 MOV [BX], DL ;低四位存入STORE中 INC BX SHR AX, CL ;右移四次 DEC CH JNZ A10 ;循环四次完了码?
B10:
MOV DL, STORE MOV CL, STORE+1 MOV BL, STORE+2 MOV AL, STORE+3 RET
;四组数分别放在AL、BL、CL和DL中
STOP:
5.4 试编写一程序,要求比较两个字符串STRING1和STRING2所含字符是否完全相同,若相同则显示
‘MATCH’, 若不相同则显示‘NO MATCH’。 答:程序如下:
DSEG SEGMENT STRING1 DB ‘I am a student.’ STRING2 DB ‘I am a student!’ YES DB ‘MATCH’, 0DH, 0AH, ‘$’ NO DB ‘NO MATCH’, 0DH, 0AH, ‘$’ DSEG ENDS
;-------------------------------------------------------------------------- CSEG SEGMENT MAIN PROC FAR
ASSUME CS: CSEG, DS: DSEG, ES: DSEG
START: PUSH DS ;设置返回DOS
SUB AX, AX PUSH AX
MOV AX, DSEG MOV DS, AX ;给DS赋值 MOV ES, AX ;给ES赋值
;
BEGIN: LEA SI, STRING1 ;设置串比较指令的初值
LEA DI, STRING2 CLD
MOV CX, STRING2 - STRING1 REPE CMPSB ;串比较 JNE DISPNO LEA DX, YES ;显示MATCH JMP DISPLAY
DISPNO: LEA DX, NO ;显示NO MATCH DISPLAY: MOV AH, 9 ;显示一个字符串的DOS调用
INT 21H RET
MAIN ENDP CSEG ENDS ;以上定义代码段
;--------------------------------------------------------------------------
END START 5.5 试编写一程序,要求能从键盘接收一个个位数N,然后响铃N次(响铃的ASCII码为07)。
答:程序段如下:
BEGIN: MOV AH, 1 ;从键盘输入一个字符的DOS调用
INT 21H SUB AL, ‘0’ JB STOP ;输入字符<‘0’吗? CMP AL, 9 ;输入字符>‘9’吗? JA STOP CBW
MOV CX, AX ;响铃次数N JCXZ STOP
BELL:
MOV DL, 07H MOV AH, 2 INT 21H CALL DELAY100ms LOOP BELL RET
;准备响铃
;显示一个字符的DOS调用,实际为响铃 ;延时100ms
STOP:
5.6 编写程序,将一个包含有20个数据的数组M分成两个数组:正数数组P和负数数组N,并分别把
这两个数组中数据的个数显示出来。 答:程序如下:
DSEG SEGMENT COUNT EQU 20 ARRAY DW 20 DUP (?) ;存放数组 COUNT1 DB 0 ;存放正数的个数 ARRAY1 DW 20 DUP (?) ;存放正数 COUNT2 DB 0 ;存放负数的个数 ARRAY2 DW 20 DUP (?) ;存放负数 ZHEN DB 0DH, 0AH, ‘The positive number is:’, ‘$’ ;正数的个数是: FU DB 0DH, 0AH, ‘The negative number is:’, ‘$’ ;负数的个数是: CRLF DB 0DH, 0AH, ‘$’ DSEG ENDS
;-------------------------------------------------------------------------- CSEG SEGMENT MAIN PROC FAR
ASSUME CS: CSEG, DS: DSEG
START: PUSH DS ;设置返回DOS
SUB AX, AX PUSH AX
MOV AX, DSEG MOV DS, AX ;给DS赋值
BEGIN: MOV CX, COUNT
LEA BX, ARRAY LEA SI, ARRAY1 LEA DI, ARRAY2
BEGIN1: MOV AX, [BX]
CMP AX, 0 ;是负数码? JS FUSHU MOV [SI], AX ;是正数,存入正数数组 INC COUNT1 ;正数个数+1 ADD SI, 2 JMP SHORT NEXT
FUSHU: MOV [DI], AX ;是负数,存入负数数组
INC COUNT2 ;负数个数+1 ADD DI, 2
NEXT: ADD BX, 2
LOOP BEGIN1 LEA DX, ZHEN ;显示正数个数 MOV AL, COUNT1 CALL DISPLAY ;调显示子程序 LEA DX, FU ;显示负数个数 MOV AL, COUNT2
CALL DISPLAY ;调显示子程序 RET
MAIN ENDP
;-------------------------------------------------------------------------- DISPLAY PROC NEAR ;显示子程序
MOV AH, 9 ;显示一个字符串的DOS调用 INT 21H AAM ;将(AL)中的二进制数转换为二个非压缩BCD码 ADD AH, ‘0’ ;变为0~9的ASCII码 MOV DL, AH MOV AH, 2 ;显示一个字符的DOS调用 INT 21H
ADD AL, ‘0’ ;变为0~9的ASCII码 MOV DL, AL MOV AH, 2 ;显示一个字符的DOS调用 INT 21H LEA DX, CRLF ;显示回车换行 MOV AH, 9 ;显示一个字符串的DOS调用 INT 21H RET
DISPLAY ENDP ;显示子程序结束 CSEG ENDS ;以上定义代码段
;--------------------------------------------------------------------------
END START
5.7 试编写一个汇编语言程序,求出首地址为DATA的100D字数组中的最小偶数,并把它存放在AX中。
答:程序段如下:
BEGIN: MOV BX, 0
MOV CX, 100
COMPARE: MOV AX, DATA[BX] ;取数组的第一个偶数
ADD BX, 2 TEST AX, 01H ;是偶数吗? LOOPNZ COMPARE ;不是,比较下一个数 JNZ STOP ;没有偶数,退出 JCXZ STOP ;最后一个数是偶数,即为最小偶数,退出
COMPARE1: MOV DX, DATA[BX] ;取数组的下一个偶数
ADD BX, 2 TEST DX, 01H ;是偶数吗? JNZ NEXT ;不是,比较下一个数 CMP AX, DX ;(AX)<(DX)吗? JLE NEXT MOV AX, DX ;(AX)<(DX),则置换(AX)为最小偶数
NEXT: LOOP COMPARE1 STOP: RET 5.8 把AX中存放的16位二进制数K看作是8个二进制的“四分之一字节”。试编写程序要求数一下值为3(即11B)的四分之一字节数,并将该数(即11B的个数)在终端上显示出来。 答:程序段如下:
BEGIN: MOV DL, 0 ;计数初始值
MOV CX, 8
COMPARE: TEST AX, 03H ;是数03吗?
JNZ NOEQUAL ;不是,转走 INC DL ;是,计数
NOEQUAL:
ROR AX, 1 ROR AX, 1 LOOP COMPARE ADD DL, ‘0’ MOV AH, 2 INT 21H RET
;准备判断下一个数
;将计数值转换为ASCII码 ;进行显示
STOP:
5.9 试编写一个汇编语言程序,要求从键盘接收一个四位的16进制数,并在终端上显示与它等值的二
进制数。
答:程序段如下:
BEGIN: MOV BX, 0 ;用于存放四位的16进制数
MOV CH, 4 MOV CL, 4
INPUT: SHL BX, CL ;将前面输入的数左移4位
MOV AH, 1 ;从键盘取数 INT 21H CMP AL, 30H ;<0吗? JB INPUT ;不是‘0~F’的数重新输入 CMP AL, 39H ;是‘0~9’吗? JA AF ;不是,转‘A~F’的处理 AND AL, 0FH ;转换为:0000B~1001B JMP BINARY
AF: AND AL, 1101 1111B ;转换为大写字母
CMP AL, 41H ;又F吗? JA INPUT ;不是‘A~F’的数重新输入 AND AL, 0FH ;转换为:1010B~1111B ADD AL, 9
BINARY: OR BL, AL ;将键盘输入的数进行组合
DEL CH JNZ INPUT
DISPN: MOV CX, 16 ;将16位二进制数一位位地转换成ASCII码显示 DISP: MOV DL, 0
ROL BX, 1 RCL DL, 1 OR DL, 30H MOV AH, 2 ;进行显示 INT 21H LOOP DISP
STOP: RET 5.10 设有一段英文,其字符变量名为ENG,并以$字符结束。试编写一程序,查对单词SUN在该文中的出现次数,并以格式“SUN:xxxx”显示出次数。 答:程序如下:
DSEG SEGMENT ENG DB ‘Here is sun, sun ,…,$’ DISP DB ‘SUN:’ DAT DB ‘0000’ , 0DH, 0AH, ‘$’ KEYWORD DB ‘sun’ DSEG ENDS
;-------------------------------------------------------------------------- CSEG SEGMENT
MAIN START:
PROC FAR
ASSUME CS: CSEG, DS: DSEG, ES: DSEG PUSH DS ;设置返回DOS SUB AX, AX PUSH AX
MOV AX, DSEG MOV DS, AX ;给DS赋值 MOV ES, AX ;给ES赋值
MOV AX, 0
MOV DX, DISP-ENG-2 ;计算ENG的长度(每次比较sun,因此比较次数-2) LEA BX, ENG
COMP: MOV DI, BX
LEA SI, KEYWORD MOV CX, 3 REPE CMPSB ;串比较 JNZ NOMATCH INC AX ;是,SUN的个数加1 ADD BX, 2
NOMATCH: INC BX ;指向ENG的下一个字母
DEC DX JNZ COMP
DONE: MOV CH, 4 ;将次数转换为16进制数的ASCII码
MOV CL, 4 LEA BX, DAT ;转换结果存入DAT单元中
DONE1: ROL AX, CL
MOV DX, AX AND DL, 0FH ;取一位16进制数 ADD DL, 30H CMP DL, 39H JLE STORE ADD DL, 07H ;是“A~F”所以要加7
STORE: MOV [BX], DL ;转换结果存入DAT单元中
INC BX DEC CH JNZ DONE1
DISPLAY: LEA DX, DISP ;显示字符串程序(将DISP和DAT一起显示)
MOV AH, 09H INT 21H RET
MAIN ENDP CSEG ENDS ;以上定义代码段
;--------------------------------------------------------------------------
END START
5.11 从键盘输入一系列以$为结束符的字符串,然后对其中的非数字字符计数,并显示出计数结果。
答:程序段如下:
DSEG SEGMENT BUFF DB 50 DUP (‘ ’) COUNT DW 0 DSEG ENDS
┇
BEGIN: LEA BX, BUFF
MOV COUNT, 0
BEGIN:
INPUT:
MOV AH, 01 INT 21H MOV [BX], AL INC BX
CMP AL, ‘$’ JNZ INPUT LEA BX, BUFF MOV CL, [BX] INC BX
CMP CL, ‘$’ JZ DISP CMP CL, 30H JB NEXT CMP CL, 39H JA NEXT INC COUNT JMP NEXT ┇
;从键盘输入一个字符的功能调用
;是$结束符吗?
;不是,继续输入 ;对非数字字符进行计数
NEXT:
;是$结束符,则转去显示
;小于0是非数字字符 ;大于9是非数字字符
;个数+1
;16进制数显示程序段(省略)
DISP:
5.12 有一个首地址为MEM的100D字数组,试编制程序删除数组中所有为0的项,并将后续项向前压缩,
最后将数组的剩余部分补上0。 答:程序如下:
DSEG SEGMENT MEM DW 100 DUP (?) DSEG ENDS
;-------------------------------------------------------------------------- CSEG SEGMENT MAIN PROC FAR
ASSUME CS: CSEG, DS: DSEG
START: PUSH DS ;设置返回DOS
SUB AX, AX PUSH AX
MOV AX, DSEG MOV DS, AX ;给DS赋值
BEGIN: MOV SI, (100-1)*2 ;(SI)指向MEM的末元素的首地址
MOV BX, -2 ;地址指针的初值 MOV CX, 100
COMP: ADD BX, 2
CMP MEM [BX], 0 JZ CONS LOOP COMP JMP FINISH ;比较完了,已无0则结束
CONS: MOV DI, BX CONS1: CMP DI, SI ;到了最后单元码?
JAE NOMOV
MOV AX, MEM [DI+2] ;后面的元素向前移位 MOV MEM [DI], AX ADD DI, 2 JMP CONS1
NOMOV: MOV WORD PTR [SI], 0 ;最后单元补0
LOOP COMP
FINISH: RET MAIN ENDP
CSEG ENDS ;以上定义代码段
;--------------------------------------------------------------------------
END START
5.13 在STRING到STRING+99单元中存放着一个字符串,试编制一个程序测试该字符串中是否存在数字,
如有则把CL的第5位置1,否则将该位置0。 答:程序如下:
DSEG SEGMENT
STRING DB 100 DUP (?) DSEG ENDS
;-------------------------------------------------------------------------- CSEG SEGMENT MAIN PROC FAR
ASSUME CS: CSEG, DS: DSEG
START: PUSH DS ;设置返回DOS
SUB AX, AX PUSH AX
MOV AX, DSEG MOV DS, AX ;给DS赋值
BEGIN: MOV SI, 0 ;(SI)作为地址指针的变化值
MOV CX, 100
REPEAT: MOV AL, STRING [SI]
CMP AL, 30H JB GO_ON CMP AL, 39H JA GO_ON OR CL, 20H ;存在数字把CL的第5位置1 JMP EXIT
GO_ON: INC SI
LOOP REPEAT AND CL, 0DFH ;不存在数字把CL的第5位置0
EXIT: RET MAIN ENDP CSEG ENDS ;以上定义代码段
;--------------------------------------------------------------------------
END START 5.14 在首地址为TABLE的数组中按递增次序存放着100H个16位补码数,试编写一个程序把出现次数
最多的数及其出现次数分别存放于AX和CX中。 答:程序如下:
DSEG SEGMENT TABLE DW 100H DUP (?) ;数组中的数据是按增序排列的 DATA DW ? COUNT DW 0 DSEG ENDS
;-------------------------------------------------------------------------- CSEG SEGMENT MAIN PROC FAR
ASSUME CS: CSEG, DS: DSEG
START: PUSH DS ;设置返回DOS
SUB AX, AX PUSH AX
MOV AX, DSEG
MOV DS, AX
;给DS赋值
MOV CX, 100H ;循环计数器 MOV SI, 0
NEXT: MOV DX, 0
MOV AX, TABLE [SI]
COMP: CMP TABLE [SI], AX ;计算一个数的出现次数
JNE ADDR INC DX ADD SI, 2 LOOP COMP
ADDR: CMP DX, COUNT ;此数出现的次数最多吗?
JLE DONE
MOV COUNT, DX ;目前此数出现的次数最多,记下次数 MOV DATA, AX ;记下此数
DONE: LOOP NEXT ;准备取下一个数
MOV CX, COUNT ;出现最多的次数存入(CX) MOV AX, DATA ;出现最多的数存入(AX) RET
MAIN ENDP CSEG ENDS ;以上定义代码段
;--------------------------------------------------------------------------
END START
5.15 数据段中已定义了一个有n个字数据的数组M,试编写一程序求出M中绝对值最大的数,把它放在
数据段的M+2n单元中,并将该数的偏移地址存放在M+2(n+1)单元中。 答:程序如下:
DSEG SEGMENT n EQU 100H ;假设n=100H M DW n DUP (?) DATA DW ? ;M+2n单元 ADDR DW ? ;M+2(n+1)单元 DSEG ENDS
;-------------------------------------------------------------------------- CSEG SEGMENT MAIN PROC FAR
ASSUME CS: CSEG, DS: DSEG
START: PUSH DS ;设置返回DOS
SUB AX, AX PUSH AX
MOV AX, DSEG MOV DS, AX ;给DS赋值
BEGIN: MOV CX, n ;循环计数器
LEA DI, M MOV AX, [DI] ;取第一个数 MOV ADDR, DI ;记下绝对值最大的数的地址 CMP AX, 0 ;此数是正数吗? JNS ZHEN ;是正数,即为绝对值,转去判断下一个数 NEG AX ;不是正数,变为其绝对值
ZHEN: MOV BX, [DI]
CMP BX, 0 ;此数是正数吗? JNS COMP ;是正数,即为绝对值,转去比较绝对值大小 NEG BX ;不是正数,变为其绝对值
COMP: CMP AX, BX ;判断绝对值大小
BEGIN:
JAE ADDRESS MOV AX, BX ;(AX)<(BX),使(AX)中为绝对值最大的数 MOV ADDR, DI ;记下绝对值最大的数的地址
ADDRESS: ADD DI, 2
LOOP ZHEN
MOV DATA, AX ;记下此数 RET
MAIN ENDP CSEG ENDS ;以上定义代码段
;--------------------------------------------------------------------------
END START
5.16 在首地址为DATA的字数组中存放着100H个16位补码数,试编写一个程序求出它们的平均值放在
AX寄存器中;并求出数组中有多少个数小于此平均值,将结果放在BX寄存器中。 答:程序如下:
DSEG SEGMENT DATA DW 100H DUP (?) DSEG ENDS
;-------------------------------------------------------------------------- CSEG SEGMENT MAIN PROC FAR
ASSUME CS: CSEG, DS: DSEG
START: PUSH DS ;设置返回DOS
SUB AX, AX PUSH AX
MOV AX, DSEG MOV DS, AX ;给DS赋值
BEGIN: MOV CX, 100H ;循环计数器
MOV SI, 0 MOV BX, 0 ;和((DI),(BX))的初始值 MOV DI, 0
NEXT: MOV AX, DATA [SI]
CWD
ADD BX, AX ;求和 ADC DI, DX ;加上进位位 ADD SI, 2 LOOP NEXT
MOV DX, DI ;将((DI),(BX))中的累加和放入((DX),(AX))中 MOV AX, BX MOV CX, 100H IDIV CX ;带符号数求平均值,放入(AX)中
MOV BX, 0 MOV SI, 0
COMP: CMP AX, DATA [SI] ;寻找小于平均值的数
JLE NO INC BX ;小于平均值数的个数+1
NO: ADD SI, 2
LOOP COMP RET
MAIN ENDP CSEG ENDS ;以上定义代码段
;--------------------------------------------------------------------------
END START
5.17 试编制一个程序把AX中的16进制数转换为ASCII码,并将对应的ASCII码依次存放到MEM数组
中的四个字节中。例如,当(AX)=2A49H时,程序执行完后,MEM中的4个字节内容为39H,34H,41H,32H。 答:程序如下:
DSEG SEGMENT MEM DB 4 DUP (?) N DW 2A49H DSEG ENDS
;-------------------------------------------------------------------------- CSEG SEGMENT MAIN PROC FAR
ASSUME CS: CSEG, DS: DSEG
START: PUSH DS ;设置返回DOS
SUB AX, AX PUSH AX
MOV AX, DSEG MOV DS, AX ;给DS赋值
BEGIN: MOV CH, 4 ;循环计数器
MOV CL, 4 MOV AX, N LEA BX, MEM
ROTATE: MOV DL, AL ;从最低四位开始转换为ASCII码
AND DL, 0FH ADD DL, 30H CMP DL, 3AH ;是0~9吗? JL NEXT ADD DL, 07H ;是A~F
NEXT: MOV [BX], DL ;转换的ASCII码送入MEM中
INC BX ROR AX, CL ;准备转换下一位 DEC CH JNZ ROTATE RET
MAIN ENDP CSEG ENDS ;以上定义代码段
;--------------------------------------------------------------------------
END START 5.18 把0~100D之间的30个数存入以GRADE为首地址的30字数组中,GRADE+i表示学号为i+1的学生
的成绩。另一个数组RANK为30个学生的名次表,其中RANK+i的内容是学号为i+1的学生的名次。编写一程序,根据GRADE中的学生成绩,将学生名次填入RANK数组中。(提示:一个学生的名次等于成绩高于这个学生的人数加1。) 答:程序如下:
DSEG SEGMENT GRADE DW 30 DUP (?) ;假设已预先存好30名学生的成绩 RANK DW 30 DUP (?) DSEG ENDS
;-------------------------------------------------------------------------- CSEG SEGMENT MAIN PROC FAR
ASSUME CS: CSEG, DS: DSEG
START:
PUSH DS SUB AX, AX PUSH AX
MOV AX, DSEG MOV DS, AX
;设置返回DOS
;给DS赋值
MOV DI, 0 MOV CX, 30 ;外循环计数器
LOOP1: PUSH CX
MOV CX, 30 ;内循环计数器 MOV SI, 0
MOV AX, GRADE [DI] MOV DX, 1 ;起始名次为第1名
LOOP2: CMP GRADE [SI], AX ;成绩比较
JBE GO_ON INC DX ;名次+1
GO_ON: ADD SI, 2
LOOP LOOP2 POP CX
MOV RNAK [DI], DX ;名次存入RANK数组 ADD DI, 2 LOOP LOOP1 RET
MAIN ENDP CSEG ENDS ;以上定义代码段
;--------------------------------------------------------------------------
END START
5.19 已知数组A包含15个互不相等的整数,数组B包含20个互不相等的整数。试编制一程序把既在A
中又在B中出现的整数存放于数组C中。 答:程序如下:
DSEG SEGMENT A DW 15 DUP (?) B DW 20 DUP (?) C DW 15 DUP (‘ ’) DSEG ENDS
;-------------------------------------------------------------------------- CSEG SEGMENT MAIN PROC FAR
ASSUME CS: CSEG, DS: DSEG
START: PUSH DS ;设置返回DOS
SUB AX, AX PUSH AX
MOV AX, DSEG MOV DS, AX ;给DS赋值
BEGIN: MOV SI, 0
MOV BX, 0 MOV CX, 15 ;外循环计数器
LOOP1: PUSH CX
MOV CX, 20 ;内循环计数器 MOV DI, 0
MOV AX, A [SI] ;取A数组中的一个数
LOOP2: CMP B [DI], AX ;和B数组中的数相等吗?
BEGIN:
JNE NO
MOV C [BX], AX ;相等存入C数组中 ADD BX, 2
NO: ADD DI, 2
LOOP LOOP2 ADD SI, 2 POP CX LOOP LOOP1 RET
MAIN ENDP CSEG ENDS ;以上定义代码段
;--------------------------------------------------------------------------
END START
5.20 设在A、B和C单元中分别存放着三个数。若三个数都不是0,则求出三数之和存放在D单元中;
若其中有一个数为0,则把其它两单元也清0。请编写此程序。 答:程序如下:
DSEG SEGMENT A DW ? B DW ? C DW ? D DW 0 DSEG ENDS
;-------------------------------------------------------------------------- CSEG SEGMENT MAIN PROC FAR
ASSUME CS: CSEG, DS: DSEG
START: PUSH DS ;设置返回DOS
SUB AX, AX PUSH AX
MOV AX, DSEG MOV DS, AX ;给DS赋值
BEGIN: CMP A, 0
JE NEXT CMP B, 0 JE NEXT CMP C, 0 JE NEXT MOV AX, A ADD AX, B ADD AX, C MOV D, AX JMP SHORT EXIT
NEXT: MOV A, 0
MOV B, 0 MOV C, 0
EXIT: RET MAIN ENDP CSEG ENDS ;以上定义代码段
;--------------------------------------------------------------------------
END START
试编写一程序,要求比较数组ARRAY中的三个16位补码数,并根据比较结果在终端上显示如下信息:
(1) 如果三个数都不相等则显示0; (2) 如果三个数有二个数相等则显示1; (3) 如果三个数都相等则显示2。 答:程序如下:
DSEG SEGMENT ARRAY DW 3 DUP (?) DSEG ENDS
;-------------------------------------------------------------------------- CSEG SEGMENT MAIN PROC FAR
ASSUME CS: CSEG, DS: DSEG
START: PUSH DS ;设置返回DOS
SUB AX, AX PUSH AX
MOV AX, DSEG MOV DS, AX ;给DS赋值
BEGIN: LEA SI, ARRAY
MOV DX, 0 ;(DX)用于存放所求的结果 MOV AX, [SI] MOV BX, [SI+2] CMP AX, BX ;比较第一和第二两个数是否相等 JNE NEXT1 INC DX
NEXT1: CMP [SI+4], AX ;比较第一和第三两个数是否相等
JNE NEXT2 INC DX
NEXT2: CMP [SI+4], BX ;比较第二和第三两个数是否相等
JNE NUM INC DX
NUM: CMP DX, 3
JL DISP DEC DX
DISP: ADD DL, 30H ;转换为ASCII码
MOV AH, 2 ;显示一个字符 INT 21H RET
MAIN ENDP CSEG ENDS ;以上定义代码段
;--------------------------------------------------------------------------
END START 5.21 从键盘输入一系列字符(以回车符结束),并按字母、数字、及其它字符分类计数,最后显示出这
三类的计数结果。 答:程序如下:
DSEG SEGMENT
ALPHABET DB ‘输入的字母字符个数为:’, ‘$’ NUMBER DB ‘输入的数字字符个数为:’, ‘$’ OTHER DB ‘输入的其它字符个数为:’, ‘$’ CRLF DB 0DH, 0AH, ‘$’ DSEG ENDS
;--------------------------------------------------------------------------
CSEG
SEGMENT
MAIN START:
PROC FAR
ASSUME CS: CSEG, DS: DSEG PUSH DS ;设置返回DOS SUB AX, AX PUSH AX
MOV AX, DSEG MOV DS, AX ;给DS赋值
MOV BX, 0 ;字母字符计数器 MOV SI, 0 ;数字字符计数器 MOV DI, 0 ;其它字符计数器
INPUT: MOV AH, 1 ;输入一个字符
INT 21H CMP AL, 0DH ;是回车符吗? JE DISP CMP AL, 30H ;<数字0吗? JAE NEXT1
OTHER: INC DI ;是其它字符
JMP SHORT INPUT
NEXT1: CMP AL, 39H ;>数字9吗?
JA NEXT2 INC SI ;是数字字符 JMP SHORT INPUT
NEXT2: CMP AL, 41H ;<字母A吗?
JAE NEXT3 JMP SHORT OTHER ;是其它字符
NEXT3: CMP AL, 5AH ;>字母Z吗?
JA NEXT4 INC BX ;是字母字符A~Z JMP SHORT INPUT
NEXT4: CMP AL, 61H ;<字母a吗?
JAE NEXT5 JMP SHORT OTHER ;是其它字符
NEXT5: CMP AL, 7AH ;>字母z吗?
JA SHORT OTHER ;是其它字符 INC BX ;是字母字符a~z JMP SHORT INPUT
DISP: LEA DX, ALPHABET
CALL DISPLAY LEA DX, NUMBER MOV BX, SI CALL DISPLAY LEA DX, OTHER MOV BX, DI CALL DISPLAY RET
MAIN ENDP
;-------------------------------------------------------------------------- DISPLAY PROC NEAR
MOV AH, 09H ;显示字符串功能调用 INT 21H CALL BINIHEX ;调把BX中二进制数转换为16进制显示子程序
BEGIN:
LEA DX, CRLF MOV AH, 09H ;显示回车换行 INT 21H RET
DISPLAY ENDP
;-------------------------------------------------------------------------- BINIHEX PROC NEAR ;将BX中二进制数转换为16进制数显示子程序
MOV CH, 4
ROTATE: MOV CL, 4
ROL BX, CL MOV DL, BL AND DL, 0FH ADD DL, 30H CMP DL, 3AH ;是A~F吗? JL PRINT_IT ADD DL, 07H
PRINT_IT: MOV AH, 02H ;显示一个字符
INT 21H DEC CH JNZ ROTATE RET
BINIHEX ENDP CSEG ENDS ;以上定义代码段
;--------------------------------------------------------------------------
END START
5.22 已定义了两个整数变量A和B,试编写程序完成下列功能:
(1) 若两个数中有一个是奇数,则将奇数存入A中,偶数存入B中; (2) 若两个数中均为奇数,则将两数加1后存回原变量; (3) 若两个数中均为偶数,则两个变量均不改变。 答:程序如下:
DSEG SEGMENT A DW ? B DW ? DSEG ENDS
;-------------------------------------------------------------------------- CSEG SEGMENT MAIN PROC FAR
ASSUME CS: CSEG, DS: DSEG
START: PUSH DS ;设置返回DOS
SUB AX, AX PUSH AX
MOV AX, DSEG MOV DS, AX ;给DS赋值
BEGIN: MOV AX, A
MOV BX, B XOR AX, BX TEST AX, 0001H ;A和B同为奇数或偶数吗? JZ CLASS ;A和B都为奇数或偶数,转走 TEST BX, 0001H JZ EXIT ;B为偶数,转走 XCHG BX, A ;A为偶数,将奇数存入A中 MOV B, BX ;将偶数存入B中
JMP EXIT
CLASS: TEST BX, 0001H ;A和B都为奇数吗?
JZ EXIT ;A和B同为偶数,转走 INC B INC A
EXIT: RET MAIN ENDP CSEG ENDS ;以上定义代码段
;--------------------------------------------------------------------------
END START
5.23 假设已编制好5个歌曲程序,它们的段地址和偏移地址存放在数据段的跳跃表SINGLIST中。试编
制一程序,根据从键盘输入的歌曲编号1~5,转去执行五个歌曲程序中的某一个。 答:程序如下:
DSEG SEGMENT
SINGLIST DD SING1
DD SING2 DD SING3 DD SING4 DD SING5
ERRMSG DB ‘Error! Invalid parameter!’, 0DH, 0AH, ‘$’ DSEG ENDS
;-------------------------------------------------------------------------- CSEG SEGMENT MAIN PROC FAR
ASSUME CS: CSEG, DS: DSEG
START: PUSH DS ;设置返回DOS
SUB AX, AX PUSH AX
MOV AX, DSEG MOV DS, AX ;给DS赋值
BEGIN: MOV AH, 1 ;从键盘输入的歌曲编号1~5
INT 21H CMP AL, 0DH JZ EXIT ;是回车符,则结束 SUB AL, ‘1’ ;是1~5吗? JB ERROR ;小于1,错误 CMP AL, 4 JA ERROR ;大于5,错误 MOV BX, OFFSET SINGLIST MUL AX, 4 ;(AX)=(AL)*4,每个歌曲程序的首地址占4个字节 ADD BX, AX JMP DWORD PTR[BX] ;转去执行歌曲程序
ERROR: MOV DX, OFFSET ERRMSG
MOV AH, 09H INT 21H ;显示错误信息 JMP BEGIN
SING1: ┇
JMP BEGIN
SING2: ┇
JMP BEGIN
SING3: ┇
JMP BEGIN
┇ JMP BEGIN
SING5: ┇
JMP BEGIN
EXIT: RET MAIN ENDP CSEG ENDS ;以上定义代码段
;--------------------------------------------------------------------------
END START
5.24 试用8086的乘法指令编制一个32位数和16位数相乘的程序;再用80386的乘法指令编制一个32
位数和16位数相乘的程序,并定性比较两个程序的效率。 答:8086的程序如下(假设为无符号数):
DSEG SEGMENT MUL1 DD ? ;32位被乘数 MUL2 DW ? ;16位乘数 MUL0 DW 0,0 ,0 ,0 ;乘积用64位单元存放 DSEG ENDS
;-------------------------------------------------------------------------- CSEG SEGMENT MAIN PROC FAR
ASSUME CS: CSEG, DS: DSEG
START: PUSH DS ;设置返回DOS
SUB AX, AX PUSH AX
MOV AX, DSEG MOV DS, AX ;给DS赋值
BEGIN: MOV BX, MUL2 ;取乘数
MOV AX, WORD PTR MUL1 ;取被乘数低位字 MUL BX
MOV MUL0, AX ;保存部分积低位 MOV MUL0+2, DX ;保存部分积高位 MOV AX, WORD PTR[MUL1+2] ;取被乘数高位字 MUL BX
ADD MUL0+2, AX ;部分积低位和原部分积高位相加 ADC MUL0+4, DX ;保存部分积最高位,并加上进位
EXIT: RET MAIN ENDP CSEG ENDS ;以上定义代码段
;--------------------------------------------------------------------------
END START
80386的程序如下(假设为无符号数):
.386 DSEG SEGMENT MUL1 DD ? ;32位被乘数 MUL2 DW ? ;16位乘数 MUL0 DD 0,0 ;乘积用64位单元存放 DSEG ENDS
;-------------------------------------------------------------------------- CSEG SEGMENT MAIN PROC FAR
ASSUME CS: CSEG, DS: DSEG
SING4:
START:
PUSH DS SUB AX, AX PUSH AX
MOV AX, DSEG MOV DS, AX
;设置返回DOS
;给DS赋值
MOVZX EBX, MUL2 ;取乘数,并0扩展成32位 MOV EAX, MUL1 ;取被乘数 MUL EBX
MOV DWORD PTR MUL0, EAX ;保存积的低位双字 MOV DWORD PTR[MUL0+4], EDX ;保存积的高位双字
EXIT: RET MAIN ENDP CSEG ENDS ;以上定义代码段
;--------------------------------------------------------------------------
END START 80386作32位乘法运算用一条指令即可完成,而8086则需用部分积作两次完成。
5.25 如数据段中在首地址为MESS1的数据区内存放着一个长度为35的字符串,要求把它们传送到附加
段中的缓冲区MESS2中去。为提高程序执行效率,希望主要采用MOVSD指令来实现。试编写这一程序。
答:80386的程序如下:
.386
.MODEL SMALL .STACK 100H .DATA
MESS1 DB ‘123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ’,? ;长度为35的字符串
.FARDATA
MESS2 DB 36 DUP (?)
.CODE
START: MOV AX, @DATA
MOV DS, AX ;给DS赋值 MOV AX, @FARDATA MOV ES, AX ;给ES赋值 ASSUME ES:@FARDATA
BEGIN: LEA ESI, MESS1
LEA EDI, MESS2 CLD
MOV ECX, (35+1)/4 ;取传送的次数 REP MOVSD
;--------------------------------------------------------------------------
MOV AX, 4C00H ;返回DOS INT 21H END START 5.26 试用比例变址寻址方式编写一386程序,要求把两个64位整数相加并保存结果。
答:80386的程序如下:
.386
.MODEL SMALL .STACK 100H .DATA
DATA1 DQ ? DATA2 DQ ?
BEGIN:
.CODE
START: MOV AX, @DATA
MOV DS, AX ;给DS赋值
BEGIN: MOV ESI, 0
MOV EAX, DWORD PTR DATA2[ESI*4] ADD DWORD PTR DATA1[ESI*4], EAX INC ESI
MOV EAX, DWORD PTR DATA2[ESI*4] ADC DWORD PTR DATA1[ESI*4], EAX
;--------------------------------------------------------------------------
MOV AX, 4C00H ;返回DOS INT 21H END START
第 六 章. 习 题
6.1 下面的程序段有错吗?若有,请指出错误。
CRAY PROC
PUSH AX ADD AX, BX RET
ENDP CRAY
答:程序有错。改正如下:
CRAY PROC
ADD AX, BX RET
CRAY ENDP ;CRAY是过程名,应放在ENDP的前面
6.2 已知堆栈寄存器SS的内容是0F0A0H,堆栈指示器SP的内容是00B0H,先执行两条把8057H和0F79BH分别入栈的PUSH指令,然后执行一条POP指令。试画出示意图说明堆栈及SP内容的变化过程。 答:变化过程如右图所示:
9BH 再PUSH,SP-2 6.3 分析下面的程序,画出堆栈最满时各单元的地址及内容。 F7H POP, SP+2 ;******************************************** 57H 先PUSH,SP-2 80H S_SEG SEGMENT AT 1000H ;定义堆栈段
F0A0:00B0H DW 200 DUP (?) ;200*2=190H
6.2 题堆栈及SP内容的变化过程 TOS LABEL WORD
S_SEG ENDS
;******************************************** C_SEG SEGMENT ;定义代码段
ASSUME CS: C_SEG, SS: S_SEG
START: MOV AX, S_SEG
MOV SS, AX 1000:0184 MOV SP, OFFSET TOS :0186 (FLAGS) :0188 (AX)
:018A T_ADDR PUSH DS
:018C 0000 MOV AX, 0
:018E (DS) PUSH AX
┇
0186 SP:
PUSH T_ADDR
6.3 题堆栈最满时各PUSH AX
单元的地址及内容
PUSHF ┇ POPF POP AX POP T_ADDR RET
;-------------------------------------------------------------------------- C_SEG ENDS ;代码段结束 ;******************************************
END START ;程序结束
答:堆栈最满时各单元的地址及内容如右图所示:
6.4 分析下面的程序,写出堆栈最满时各单元的地址及内容。
;******************************************** STACK SEGMENT AT 500H ;定义堆栈段
DW 128 DUP (?)
TOS LABEL WORD STACK ENDS
;******************************************** CODE SEGMENT ;定义代码段 MAIN PROC FAR ;主程序部分
ASSUME CS: CODE, SS: STACK
START: MOV AX, STACK
MOV SS, AX
MOV SP, OFFSET TOS PUSH DS SUB AX, AX PUSH AX
;MAIN PART OF PROGRAM GOES HERE
MOV AX, 4321H CALL HTOA RET
MAIN ENDP ;主程序部分结束
;-------------------------------------------------------------------- HTOA PROC NEAR ;HTOA子程序
CMP AX, 15 JLE B1 PUSH AX PUSH BP
0500:00EC MOV BP, SP
:00EE 返回POP BP地址 MOV BX, [BP+2] 0003H :00F0 AND BX, 000FH :00F2 返回POP BP地址 MOV [BP+2], BX 0002H :00F4 POP BP :00F6 返回POP BP地址 MOV CL, 4 0001H :00F8 SHR AX, CL :00FA 主程序返回地址 0000 CALL HTOA :00FC POP BP (DS) :00FE B1: ADD AL, 30H
00EE SP: CMP AL, 3AH
6.4 题堆栈最满时各单元JL PRINTIT
的地址及内容 ADD AL, 7H
PRINTIT: MOV DL, AL
MOV AH, 2
INT 21H RET
HOTA ENDP ;HOTA子程序结束
;-------------------------------------------------------------------- CODE ENDS ;代码段结束
;******************************************
END START ;程序结束
答:堆栈最满时各单元的地址及内容如右上图所示:
6.5 下面是一个程序清单,请在下面的图中填入此程序执行过程中的堆栈变化。
;*************************
0000 STACKSG SEGMENT 0000 20 [. DW 32 DUP (?)
? ? ? ?
]
0040 STACKSG ENDS
;*************************
0000 CODESG SEGMENT PARA ‘CODE’
;--------------------------------------
0000 BEGIN PROC FAR
ASSUME CS: CODESG, SS: STACKSG
0000 1E PUSH DS 0001 2B C0 SUB AX, AX 0003 50 PUSH AX 0004 E8 0008 R CALL B10
;--------------------------------------
0007 CB RET 0008 BEGIN ENDP
;--------------------------------------
0008 B10 PROC 0008 E8 000C R CALL C10
;--------------------------------------
000B C3 RET 000C B10 ENDP
;--------------------------------------
000C C10 PROC
;--------------------------------------
000C C3 RET 000D C10 ENDP
;--------------------------------------
000D CODESG ENDS
;*************************
END BEGIN
答:程序执行过程中的堆栈变化如下图所示。
偏移地址 堆栈 ( 0016H ) ( 0016H ) ( 0016H ) ( 0016H )
( 0018H ) ( 0018H ) ( 0018H ) ( 0018H ) ( 001AH ) ( 001AH ) ( 001AH ) ( 001AH ) 0007 ( 001CH ) ( 001CH ) ( 001CH ) 0000 ( 001CH ) 0000 ( 001EH ) ( 001EH ) (DS) ( 001EH ) (DS) ( 001EH ) (DS) ( 0020H ) ( 0020H ) ( 0020H ) ( 0020H ) SP: 0020H 001EH 001CH 001AH BEGIN PUSH DS PUSH AX CALL B10
( 0016H ) ( 0016H ) ( 0016H ) ( 0016H ) ( 0018H ) 000B ( 0018H ) 000B ( 0018H ) 000B ( 0018H ) 000B ( 001AH ) 0007 ( 001AH ) 0007 ( 001AH ) 0007 ( 001AH ) 0007 ( 001CH ) 0000 ( 001CH ) 0000 ( 001CH ) 0000 ( 001CH ) 0000 ( 001EH ) (DS) ( 001EH ) (DS) ( 001EH ) (DS) ( 001EH ) (DS) ( 0020H ) ( 0020H ) ( 0020H ) ( 0020H )
6.6 写一段子程序SKIPLINES,完成输出空行的功能。空出的行数在AX寄存器中。
答:程序如下:
CSEG SEGMENT SKIPLINES PROC FAR
ASSUME CS: CSEG
BEGIN: PUSH CX
PUSH DX MOV CX, AX
DISP: MOV DL, 0DH ;显示回车换行,即输出空行
MOV AH, 2 ;显示一个字符的DOS调用 INT 21H MOV DL, 0AH MOV AH, 2 ;显示一个字符的DOS调用 INT 21H LOOP DISP POP DX POP CX RET
SKIPLINES ENDP
END
6.7 设有10个学生的成绩分别是76,69,84,90,73,88,99,63,100和80分。试编制一个子程
序统计60~69分,70~79分,80~89分,90~99分和100分的人数,分别存放到S6,S7,S8,S9和S10单元中。 答:程序如下:
DSEG SEGMENT
RECORD DW 76,69,84,90,73,88,99,63,100,80 S6 DW 0 S7 DW 0 S8 DW 0 S9 DW 0 S10 DW 0 DSEG ENDS
;****************************************** CSEG SEGMENT MAIN PROC FAR
ASSUME CS: CSEG, DS: DSEG
START: PUSH DS ;设置返回DOS
SUB AX, AX PUSH AX
MOV AX, DSEG MOV DS, AX ;给DS赋值
BEGIN: MOV CX, 10
CALL COUNT ┇ ;后续程序
RET
MAIN ENDP
;-------------------------------------------------------------------------- COUNT PROC NEAR ;成绩统计子程序
MOV SI, 0
NEXT: MOV AX, RECORD[SI]
MOV BX, 10 ;以下5句是根据成绩计算相对S6的地址变化量 DIV BL ;计算公式为:((成绩)/10-6)*2送(BX) MOV BL, AL ;此时(BH)保持为0不变 SUB BX, 6 ;应为只统计60分以上成绩 SAL BX, 1 ;(BX)*2 INC S6[BX] ;S6是S6,S7,S8,S9和S10单元的首地址 ADD SI, 2 LOOP NEXT RET
COUNT ENDP ;COUNT子程序结束
;-------------------------------------------------------------------------- CSEG ENDS ;以上定义代码段 ;******************************************
END START
6.8 编写一个有主程序和子程序结构的程序模块。子程序的参数是一个N字节数组的首地址TABLE,数
N及字符CHAR。要求在N字节数组中查找字符CHAR,并记录该字符出现的次数。主程序则要求从键盘接收一串字符以建立字节数组TABLE,并逐个显示从键盘输入的每个字符CHAR以及它在TABLE数组中出现的次数。(为简化起见,假设出现次数≤15,可以用16进制形式把它显示出来。) 答:程序如下:
DSEG SEGMENT TABLE DB 255 DUP (?) N DW 255 CHAR DB ? CHAR_N DB 0 ;用于记录CHAR出现的次数 CRLF DB 0DH, 0AH, ‘$’ DSEG ENDS ;以上定义数据段 ;****************************************** STACK SEGMENT
DW 100 DUP (?)
TOS LABEL WORD STACK ENDS ;以上定义堆栈段 ;****************************************** CSEG SEGMENT MAIN PROC FAR
ASSUME CS: CSEG, DS: DSEG, SS: STACK
START: MOV AX, STACK
MOV SS, AX ;给SS赋值 MOV SP, OFFSET TOS ;给SP赋值 PUSH DS ;设置返回DOS SUB AX, AX PUSH AX
MOV AX, DSEG MOV DS, AX ;给DS赋值
BEGIN: MOV BX, 0
MOV CX, 255 ;最多输入255个字符
MOV AH, 1 ;从键盘接收一个字符的DOS功能调用 INT 21H CMP AL, 0DH ;输入回车符结束输入 JZ IN_N
MOV TABLE [BX], AL INC BX LOOP INPUT
IN_N: MOV N, BX ;TABLE数组中的字符个数送N
CALL DISP_CRLF
IN_CHAR: MOV AH, 1 ;从键盘接收一个字符并回显的DOS功能调用
INT 21H CMP AL, 0DH ;输入回车符结束 JZ EXIT MOV CHAR, AL ;输入的字符存入CHAR单元 CALL SEARCH ;调搜索字符子程序 MOV DL, ‘:’ ;显示“:”,在字符CHAR(输入时回显)的后面 MOV AH, 2 ;显示一个字符 INT 21H
MOV DL, CHAR_N ;再显示CHAR出现的次数(次数≤15) AND DL, 0FH ADD DL, 30H CMP DL, 39H JBE NEXT ADD DL, 07H ;是A~F
NEXT: MOV AH, 2 ;显示一个字符
INT 21H CALL DISP_CRLF JMP SHORT IN_CHAR
EXIT: RET MAIN ENDP
;-------------------------------------------------------------------------- SEARCH PROC NEAR ;搜索字符子程序
MOV SI, 0 MOV CX, N MOV CHAR_N, 0 MOV AL, CHAR
ROTATE: CMP AL, TABLE [SI]
JNZ ROTATE1 INC CHAR_N ;搜索到字符,则出现次数+1
ROTATE1: INC SI
LOOP ROTATE RET
SEARCH ENDP ;SEARCH子程序结束
;-------------------------------------------------------------------------- DISP_CRLF PROC NEAR ;显示回车换行符子程序
LEA DX, CRLF MOV AH, 09H INT 21H RET
DISP_CRLF ENDP ;DISP_CRLF子程序结束
;-------------------------------------------------------------------------- CSEG ENDS ;以上定义代码段 ;******************************************INPUT:
END START
6.9 编写一个子程序嵌套结构的程序模块,分别从键盘输入姓名及8个字符的电话号码,并以一定的
格式显示出来。 主程序TELIST:
? 显示提示符“INPUT NAME:”; ? 调用子程序INPUT_NAME输入姓名;
? 显示提示符“INPUT A TELEPHONE NUMBER:”; ? 调用子程序INPHONE输入电话号码;
? 调用子程序PRINTLINE显示姓名及电话号码。 子程序INPUT_NAME:
? 调用键盘输入子程序GETCHAR,把输入的姓名存放在INBUF缓冲区中; ? 把INBUF中的姓名移入输出行OUTNAME。 子程序INPHONE:
? 调用键盘输入子程序GETCHAR,把输入的8位电话号码存放在INBUF缓冲区中; ? 把INBUF中的号码移入输出行OUTPHONE。 子程序PRINTLINE:
显示姓名及电话号码,格式为:
NAME TEL. X X X XXXXXXXX
答:程序如下:
DSEG SEGMENT INBUF DB 12 DUP (‘ ’) ;输入缓冲区,初始值为空格 OUTNAME DB 16 DUP (‘ ’), ;姓名输出行,初始值为空格
OUTPHONE DB 12 DUP (‘ ’), 0DH, 0AH, ‘$’ ;号码输出行,初始值为空格 MESG1 DB ‘INPUT NAME:’, ‘$’ MESG2 DB ‘INPUT A TELEPHONE NUMBER:’, ‘$’ MESG3 DB ‘NAME’, 12 DUP (‘ ’), ‘TEL.’, 0DH, 0AH, ‘$’ CRLF DB 0DH, 0AH, ‘$’ DSEG ENDS ;以上定义数据段 ;****************************************** STACK SEGMENT
DW 100 DUP (?)
TOS LABEL WORD STACK ENDS ;以上定义堆栈段 ;****************************************** CSEG SEGMENT TELIST PROC FAR ;主程序TELIST
ASSUME CS: CSEG, DS: DSEG, ES: DSEG, SS: STACK
START: MOV AX, STACK
MOV SS, AX ;给SS赋值 MOV SP, OFFSET TOS ;给SP赋值 PUSH DS ;设置返回DOS SUB AX, AX PUSH AX
MOV AX, DSEG MOV DS, AX ;给DS赋值 MOV ES, AX ;给ES赋值
BEGIN: LEA DX, MESG1
MOV AH, 09H ;显示字符串功能调用 INT 21H CALL INPUT_NAME ;输入姓名 LEA DX, MESG2
MOV AH, 09H ;显示字符串功能调用 INT 21H CALL INPHONE ;输入电话号码 CALL PRINTLINE ;显示姓名及电话号码 RET
TELIST ENDP
;-------------------------------------------------------------------------- INPUT_NAME PROC NEAR ;输入姓名子程序
CALL GETCHAR ;调输入字符子程序输入姓名 LEA SI, INBUF ;把INBUF中的姓名移入输出行OUTNAME LEA DI, OUTNAME MOV CX, 12 CLD REP MOVSB RET
INPUT_NAME ENDP ;INPUT_NAME子程序结束
;-------------------------------------------------------------------------- INPHONE PROC NEAR ;输入电话号码子程序
CALL GETCHAR ;调输入字符子程序输入电话号码 LEA SI, INBUF ;把INBUF中的电话号码移入输出行OUTPHONE LEA DI, OUTPHONE MOV CX, 12 CLD REP MOVSB RET
INPHONE ENDP ;INPHONE子程序结束
;-------------------------------------------------------------------------- GETCHAR PROC NEAR ;键盘输入子程序
MOV AL, 20H ;先将INBUF中填满空格字符 MOV CX, 12 LEA DI, INBUF CLD REP STOSB MOV CX, 12 ;向INBUF输入字符 MOV DI, 0
INPUT: MOV AH, 1 ;从键盘接收一个字符并回显的DOS功能调用
INT 21H CMP AL, 0DH ;输入回车符返回 JZ QUIT
MOV INBUF[DI], AL INC DI LOOP INPUT
QUIT: CALL DISP_CRLF
RET
GETCHAR ENDP ;GETCHAR子程序结束
;-------------------------------------------------------------------------- PRINTLINE PROC NEAR ;显示姓名及电话号码子程序
LEA DX, MESG3 MOV AH, 09H ;显示字符串功能调用 INT 21H LEA DX, OUTNAME ;显示姓名及电话号码 MOV AH, 09H ;显示字符串功能调用 INT 21H
RET
PRINTLINE ENDP ;PRINTLINE子程序结束
;-------------------------------------------------------------------------- DISP_CRLF PROC NEAR ;显示回车换行符子程序
LEA DX, CRLF MOV AH, 09H INT 21H RET
DISP_CRLF ENDP ;DISP_CRLF子程序结束
;-------------------------------------------------------------------------- CSEG ENDS ;以上定义代码段 ;******************************************
END START
6.10 编写子程序嵌套结构的程序,把整数分别用二进制和八进制形式显示出来。
主程序BANDO:把整数字变量VAL1存入堆栈,并调用子程序PAIRS;
子程序PAIRS:从堆栈中取出VAL1;调用二进制显示程序OUTBIN显示出与其等效的二进制数;输
出8个空格;调用八进制显示程序OUTOCT显示出与其等效的八进制数;调用输出回车及换行符子程序。
答:程序如下:
DSEG SEGMENT VAL1 DW ? CRLF DB 0DH, 0AH, ‘$’ DSEG ENDS ;以上定义数据段 ;****************************************** CSEG SEGMENT BANDO PROC FAR ;主程序BANDO
ASSUME CS: CSEG, DS: DSEG
START: PUSH DS ;设置返回DOS
SUB AX, AX PUSH AX
MOV AX, DSEG MOV DS, AX ;给DS赋值
PUSH VAL1 CALL PAIRS RET
BANDO ENDP
;-------------------------------------------------------------------------- PAIRS PROC NEAR ;PAIRS子程序
PUSH BP MOV BP, SP PUSH BX
MOV BX, [BP+4] ;从堆栈中取出VAL1 CALL OUTBIN ;调用二进制显示子程序 MOV CX, 8 ;显示8个空格符
SPACE: MOV DL, ‘ ’
MOV AH, 2 INT 21H LOOP SPACE CALL OUTOCT ;调用八进制显示子程序 CALL DISP_CRLF POP BX POP BP
RET 2
PAIRS ENDP ;PAIRS子程序结束
;-------------------------------------------------------------------------- OUTBIN PROC NEAR ;二进制显示子程序
PUSH BX MOV CX, 16
ONEBIT: ROL BX, 1
MOV DX, BX AND DX, 1 OR DL, 30H ;转换为ASCII码 MOV AH, 2 INT 21H LOOP ONEBIT POP BX RET
OUTBIN ENDP ;OUTBIN子程序结束
;-------------------------------------------------------------------------- OUTOCT PROC NEAR ;八进制显示子程序
ROL BX, 1 ;16位二进制数包含6位八进制数,最高位仅1位 MOV DX, BX AND DX, 1 OR DL, 30H ;转换为ASCII码 MOV AH, 2 INT 21H MOV CX, 5 ;余下还有5位八进制数
NEXT: PUSH CX
MOV CL, 3 ;1位八进制数包含3位二进制数 ROL BX, CL MOV DX, BX AND DX, 07H OR DL, 30H ;转换为ASCII码 MOV AH, 2 INT 21H POP CX LOOP NEXT RET
OUTOCT ENDP ;OUTOCT子程序结束
;-------------------------------------------------------------------------- DISP_CRLF PROC NEAR ;显示回车换行符子程序
LEA DX, CRLF MOV AH, 09H INT 21H RET
DISP_CRLF ENDP ;DISP_CRLF子程序结束
;-------------------------------------------------------------------------- CSEG ENDS ;以上定义代码段 ;******************************************
END START
6.11 假定一个名为MAINPRO的程序要调用子程序SUBPRO,试问:
(1) MAINPRO中的什么指令告诉汇编程序SUBPRO是在外部定义的? (2) SUBPRO怎么知道MAINPRO要调用它? 答:(1) EXTRN SUBPRO:FAR
(2) PUBLIC SUBPRO
6.12 假定程序MAINPRO和SUBPRO不在同一模块中,MAINPRO中定义字节变量QTY和字变量VALUE和
PRICE。SUBPRO程序要把VALUE除以QTY,并把商存在PRICE中。试问: (1) MAINPRO怎么告诉汇编程序外部子程序要调用这三个变量?
(2) SUBPRO怎么告诉汇编程序这三个变量是在另一个汇编语言程序定义的? 答:(1) PUBLIC QTY, VALUE, PRICE
(2) EXTRN QTY:BYTE, VALUE:WORD, PRICE:WORD 6.13 假设:
(1) 在模块1中定义了双字变量VAR1,首地址为VAR2的字节数据和NEAR标号LAB1,它们将由模
块2和模块3所使用;
(2) 在模块2中定义了字变量VAR3和FAR标号LAB2,而模块1中要用到VAR3,模块3中要用到
LAB2;
(3) 在模块3中定义了FAR标号LAB3,而模块2中要用到它。 试对每个源模块给出必要的EXTRN和PUBLIC说明。 答:模块1:
EXTRN VAR3: WORD
PUBLIC VAR1,VAR2,LAB1 模块2:
EXTRN VAR1: DWORD,VAR2: BYTE,LAB1: NEAR,LAB3: FAR PUBLIC VAR3,LAB2 模块3:
EXTRN VAR1: DWORD,VAR2: BYTE,LAB1: NEAR,LAB2: FAR PUBLIC LAB3 6.14 主程序CALLMUL定义堆栈段、数据段和代码段,并把段寄存器初始化,数据段中定义变量QTY和
PRICE;代码段中将PRICE装入AX,QTY装入BX,然后调用子程序SUBMUL。程序SUBMUL没有定义任何数据,它只简单地把AX中的内容(PRICE)乘以BX中的内容(QTY),乘积放在DX: AX中。请编制这两个要连接起来的程序。 答:程序如下:
TITLE CALLMUL ;主程序 EXTRN SUBMUL: FAR
;----------------------------------------------------------------- STACK SEGMENT PARA STACK ‘STACK’
DW 64 DUP (?)
TOS LABEL WORD STACK ENDS
;-------------------------------------------------------------- DATASG SEGMENT PARA ‘DATA’ QTY DW 0140H PRICE DW 2500H DATASG ENDS
;-------------------------------------------------------------- CODESG SEGMENT PARA ‘CODE’ CALLMUL PROC FAR
ASSUME CS: CODESG, DS: DATASG, SS: STACK
START: MOV AX, STACK
MOV SS, AX ;给SS赋值 MOV SP, OFFSET TOS ;给SP赋值 PUSH DS SUB AX, AX POP AX MOV AX, DATASG MOV DS, AX
MOV AX, PRICE MOV BX, QTY CALL SUBMUL RET
CALLMUL ENDP CODESG ENDS
;-----------------------------------------------------------------
END CALLMUL
;*************************************************************** TITLE SUBMUL ;子程序 PUBLIC SUBMUL
;----------------------------------------------------------------- CODESG1 SEGMENT PARA ‘CODE’
ASSUME CS: CODESG1
SUBMUL PROC FAR
ASSUME CS: CODESG1 MUL BX RET
SUBMUL ENDP CODESG1 ENDS
;-----------------------------------------------------------------
END
6.15 试编写一个执行以下计算的子程序COMPUTE:
R ← X + Y - 3
其中X,Y及R均为字数组。假设COMPUTE与其调用程序都在同一代码段中,数据段D_SEG中包含X和Y数组,数据段E_SEG中包含R数组,同时写出主程序调用COMPUTE过程的部分。
如果主程序和COMPUTE在同一程序模块中,但不在同一代码段中,程序应如何修改? 如果主程序和COMPUTE不在同一程序模块中,程序应如何修改? 答:(1) 主程序和COMPUTE在同一代码段中的程序如下:
TITLE ADDITION ;主程序
;-------------------------------------------------------------- D_SEG SEGMENT PARA ‘DATA’ COUNT EQU 10H X DW COUNT DUP (?) Y DW COUNT DUP (?) D_SEG ENDS
;-------------------------------------------------------------- E_SEG SEGMENT PARA ‘DATA’ R DW COUNT DUP (?) E_SEG ENDS
;-------------------------------------------------------------- C_SEG SEGMENT PARA ‘CODE’ ADDITION PROC FAR
ASSUME CS: C_SEG, DS: D_SEG, ES: E_SEG
START: PUSH DS
SUB AX, AX PUSH AX
MOV AX, D_SEG MOV DS, AX MOV AX, E_SEG MOV ES, AX CALL COMPUTE ;调用求和子程序
RET
ADDITION ENDP
;******************************************** COMPUTE PROC NEAR ;同一段的求和子程序
MOV CX, COUNT MOV BX, 0
REPEAT: MOV AX, X[BX]
ADD AX, Y[BX] SUB AX, 3 MOV ES: R[BX], AX RET
COMPUTE ENDP
;----------------------------------------------------------------- C_SEG ENDS
;*******************************************
END START
(2) 主程序和COMPUTE在同一程序模块中,但不在同一代码段中的程序如下:
TITLE ADDITION ;主程序
;-------------------------------------------------------------- D_SEG SEGMENT PARA ‘DATA’ COUNT EQU 10H X DW COUNT DUP (?) Y DW COUNT DUP (?) D_SEG ENDS
;-------------------------------------------------------------- E_SEG SEGMENT PARA ‘DATA’ R DW COUNT DUP (?) E_SEG ENDS
;-------------------------------------------------------------- C_SEG SEGMENT PARA ‘CODE’ ADDITION PROC FAR
ASSUME CS: C_SEG, DS: D_SEG, ES: E_SEG
START: PUSH DS
SUB AX, AX POP AX MOV AX, D_SEG MOV DS, AX MOV AX, E_SEG MOV ES, AX CALL FAR PTR COMPUTE ;调用求和子程序 RET
ADDITION ENDP
C_SEG ENDS
;******************************************** CODESG SEGMENT PARA ‘CODE’
ASSUME CS: CODESG
COMPUTE PROC FAR ;不同段的求和子程序
MOV CX, COUNT MOV BX, 0
REPEAT: MOV AX, X[BX]
ADD AX, Y[BX] SUB AX, 3 MOV ES: R[BX], AX RET
COMPUTE ENDP
;----------------------------------------------------------------- CODESG ENDS
;********************************************
END START
(3) 主程序和COMPUTE不在同一程序模块中的程序如下:
TITLE ADDITION ;主程序 EXTRN COMPUTE: FAR PUBLIC COUNT, X, Y, R
;-------------------------------------------------------------- D_SEG SEGMENT PARA ‘DATA’ COUNT DW 10H X DW 10H DUP (?) Y DW 10H DUP (?) D_SEG ENDS
;-------------------------------------------------------------- E_SEG SEGMENT PARA ‘DATA’ R DW 10H DUP (?) E_SEG ENDS
;-------------------------------------------------------------- C_SEG SEGMENT PARA ‘CODE’ ADDITION PROC FAR
ASSUME CS: C_SEG, DS: D_SEG, ES: E_SEG
START: PUSH DS
SUB AX, AX POP AX MOV AX, D_SEG MOV DS, AX MOV AX, E_SEG MOV ES, AX CALL FAR PTR COMPUTE ;调用求和子程序 RET
ADDITION ENDP C_SEG ENDS
;-----------------------------------------------------------------
END START
;***************************************************************
TITLE COMPUTE ;求和子程序 EXTRN COUNT:WORD, X:WORD, Y:WORD, R:WORD PUBLIC COMPUTE
;----------------------------------------------------------------- CODESG SEGMENT PARA ‘CODE’
ASSUME CS: CODESG
COMPUTE PROC FAR ;不同模块的求和子程序
MOV CX, COUNT MOV BX, 0
REPEAT: MOV AX, X[BX]
ADD AX, Y[BX] SUB AX, 3 MOV ES: R[BX], AX RET
COMPUTE ENDP
;----------------------------------------------------------------- CODESG ENDS
;********************************************
END
第 七 章. 习 题
7.1 编写一条宏指令CLRB,完成用空格符将一字符区中的字符取代的工作。字符区首地址及其长度为
变元。
答:宏定义如下:
CLRB MACRO N, CFIL
MOV CX, N CLD
MOV AL, ‘ ’ ;;取空格符的ASCII码 LEA DI, CFIL REP STOSB ENDM 7.2 某工厂计算周工资的方法是每小时的工资率RATE乘以工作时间HOUR,另外每工作满10小时加奖
金3元,工资总数存放在WAG中。请将周工资的计算编写成一条宏指令WAGES,并展开宏调用:
WAGES R1, 42, SUM
答:宏定义如下:
WAGES MACRO RATE, HOUR, WAG
MOV AL, HOUR ;;计算周工资(WAG),公式为:HOUR* RATE MOV BL, RATE MUL BL
MOV WAG, AX MOV AL, HOUR ;;计算奖金存入(AX),公式为:HOUR/10的商*3 MOV AH, 0 MOV BL, 10 DIV BL MOV BL, 3 MUL BL
ADD WAG, AX ;;计算周工资总数 ENDM
宏调用:
WAGES R1, 42, SUM
宏展开:
1 MOV AL, 42 1 MOV BL, R1 1 MUL BL 1 MOV SUM, AX 1 MOV AL, 42 1 MOV AH, 0 1 MOV BL, 10 1 DIV BL 1 MOV BL, 3 1 MUL BL 1 ADD SUM, AX 7.3 给定宏定义如下:(注意:此宏指令的功能是V3←|V1-V2|)
DIF MACRO X, Y
MOV AX, X
SUB AX, Y ENDM
ABSDIF MACRO V1, V2, V3
LOCAL CONT PUSH AX DIF V1, V2 CMP AX, 0 JGE CONT NEG AX
CONT: MOV V3, AX
POP AX ENDM
试展开以下调用,并判定调用是否有效。 (1) ABSDIF P1, P2, DISTANCE
(2) ABSDIF [BX], [SI], X[DI], CX (3) ABSDIF [BX][SI], X[BX][SI], 240H (4) ABSDIF AX, AX, AX
答:(1) 宏调用 ABSDIF P1, P2, DISTANCE 的宏展开如下:此宏调用有效。
1 PUSH AX 1 DIF P1, P2 1 MOV AX, P1 1 SUB AX, P2 1 CMP AX, 0 1 JGE ??0000 1 NEG AX 1 ??0000: MOV DISTANCE, AX 1 POP AX
(2) 宏调用 ABSDIF [BX], [SI], X[DI], CX 的宏展开如下:此宏调用有效。
1 PUSH AX 1 DIF [BX], [SI] 1 MOV AX, [BX] 1 SUB AX, [SI] 1 CMP AX, 0 1 JGE ??0001 1 NEG AX 1 ??0001: MOV X[DI], AX 1 POP AX
(3) 宏调用 ABSDIF [BX][SI], X[BX][SI], 240H 的宏展开如下:此宏调用无效。
1 PUSH AX 1 DIF [BX][SI], X[BX][SI] 1 MOV AX, [BX][SI] 1 SUB AX, X[BX][SI] 1 CMP AX, 0 1 JGE ??0002 1 NEG AX 1 ??0002: MOV 240H, AX 1 POP AX
(4) 宏调用 ABSDIF AX, AX, AX 的宏展开如下:此宏调用有效但无多大意义。
1 PUSH AX 1 DIF AX, AX 1 MOV AX, AX 1 SUB AX, AX 1 CMP AX, 0
1 1 1 1
JGE ??0003
NEG AX ??0003: MOV AX, AX POP AX
7.4 试编制宏定义,要求把存储器中的一个用EOT(ASCII码04H)字符结尾的字符串传送到另一个存
储区去。
答:宏定义如下:
SEND MACRO SCHARS, DCHARS
LOCAL NEXT, EXIT PUSH AX PUSH SI MOV SI, 0
NEXT: MOV AL, SCHARS[SI]
MOV DCHARS[SI], AL CMP AL, 04H ;;是EOT字符吗? JZ EXIT INC SI JMP NEXT
EXIT: POP SI
POP AX ENDM 7.5 宏指令BIN_SUB完成多个字节数据连减的功能:
RESULT←(A-B-C-D-…)
要相减的字节数据顺序存放在首地址为OPERAND的数据区中,减数的个数存放在COUNT单元中,最后结果存入RESULT单元。请编写此宏指令。 答:宏定义如下:
BIN_SUB MACRO RESULT, A, OPERAND, COUNT
LOCAL NEXT_SUB PUSH CX PUSH BX PUSH AX
MOV CX, COUNT MOV AL, A LEA BX, OPERAND CLC
NEXT_SUB: SBB AL, [BX]
INC BX LOOP NEXT_SUB MOV RESULT, AL POP AX POP BX POP CX ENDM 7.6 请用宏指令定义一个可显示字符串GOOD: ‘GOOD STUDENTS: CLASSX NAME’,其中X和NAME在宏
调用时给出。 答:宏定义如下:
DISP_GOOD MACRO X, NAME
GOOD DB ‘GOOD STUDENTS: CLASS&X &NAME’, 0DH, 0AH, ‘$’ ENDM 7.7 下面的宏指令CNT和INC1完成相继字存储。
CNT MACRO A, B
INC1 B=B+1
ENDM 请展开下列宏调用: C=0
INC1 DATA, C INC1 DATA, C
答:宏展开如下:
C=0
INC1 DATA, C
1 DATA0 DW ?
INC1 DATA, C
1 DATA0 DW ?
A&B DW ?
ENDM
MACRO A, B
CNT A, %B
(注意:C为0没有变)
7.8 定义宏指令并展开宏调用。宏指令JOE把一串信息‘MESSAGE NO. K’存入数据存储区XK中。宏
调用为: I=0
JOE TEXT, I ┇ JOE TEXT, I ┇ JOE TEXT, I ┇
答:宏定义如下:
MARY MACRO X, K
X&K DB ‘MESSAGE NO. &K’ ENDM
JOE MACRO A, I
MARY A, %I
I=I+1
ENDM
宏调用和宏展开:
I=0
JOE TEXT, I
1 TEXT0 DB ‘MESSAGE NO. 0’
┇ JOE TEXT, I
1 TEXT1 DB ‘MESSAGE NO. 1’
┇ JOE TEXT, I
1 TEXT2 DB ‘MESSAGE NO. 2’ 7.9 宏指令STORE定义如下:
STORE MACRO X, N
MOV X+I, I
I=I+1
IF I-N STORE X, N ENDIF ENDM
试展开下列宏调用:
I=0
STORE TAB, 7
答:宏展开如下:
I=0
STORE TAB, 7
1 MOV TAB+0, 0 1 MOV TAB+1, 1 1 MOV TAB+2, 2 1 MOV TAB+3, 3 1 MOV TAB+4, 4 1 MOV TAB+5, 5 1 MOV TAB+6, 6
7.10 试编写非递归的宏指令,使其完成的工作与7.9题的STORE相同。
答:宏定义如下:
STORE MACRO K
MOV TAB+K, K ENDM
宏调用:
I=0
REPT 7
STORE %I
I=I+1
ENDM
7.11 试编写一段程序完成以下功能,如给定名为X的字符串长度大于5时,下列指令将汇编10次。
ADD AX, AX
答:程序段如下:
X DB ‘ABCDEFG’
IF ($-X) GT 5 REPT 10
ADD AX, AX
ENDM ENDIF 7.12 定义宏指令FINSUM:比较两个数X和Y(X、Y为数,而不是地址),若X>Y则执行SUM←X+2*Y;否
则执行SUM←2*X+Y。 答:宏定义如下:
CALCULATE MACRO A, B, RESULT ;;计算RESULT←2*A+B
MOV AX, A SHL AX, 1 ADD AX, B
MOV RESULT, AX ENDM
FINSUM MACRO X, Y, SUM
IF X GT Y
CALCULATE Y, X, SUM
ELSE
CALCULATE X, Y, SUM
ENDIF ENDM 7.13 试编写一段程序完成以下功能:如变元X=‘VT55’,则汇编MOV TERMINAL, 0;否则汇编
MOV TERMINAL, 1。 答:宏定义如下:
BRANCH
MACRO IFIDN ELSE
MOV TERMINAL, 1
ENDIF ENDM
7.14 对于DOS功能调用,所有的功能调用都需要在AH寄存器中存放功能码,而其中有一些功能需要在
DX中放一个值。试定义宏指令DOS21,要求只有在程序中定义了缓冲区时,汇编为:
MOV AH, DOSFUNC
MOV DX, OFFSET BUFF INT 21H
否则,无MOV DX, OFFSET BUFF指令。并展开以下宏调用:
DOS21 01
DOS21 0AH, IPFIELD
答:宏定义如下:
DOS21 MACRO DOSFUNC, BUFF
MOV AH, DOSFUNC IFDEF BUFF
MOV DX, OFFSET BUFF
ENDIF INT 21H ENDM
宏展开:
DOS21 01
1 MOV AH, 01 1 INT 21H
DOS21 0AH, IPFIELD
1 MOV AH, 0AH 1 MOV DX, OFFSET IPFIELD 1 INT 21H 7.15 编写一段程序,使汇编程序根据SIGN中的内容分别产生不同的指令。如果(SIGN)=0,则用字节变
量DIVD中的无符号数除以字节变量SCALE;如果(SIGN)=1,则用字节变量DIVD中的带符号数除以字节变量SCALE,结果都存放在字节变量RESULT中。 答:程序段如下:
MOV AL, DIVD IF SIGN
MOV AH, 0 DIV SCALE ELSE
CBW IDIV SCALE ENDIF
MOV RESULT, AL 7.16 试编写宏定义SUMMING,要求求出双字数组中所有元素之和,并把结果保存下来。该宏定义的哑元
应为数组首址ARRAY,数组长度COUNT和结果存放单元RESULT。 答:宏定义如下:
SUMMING MACRO ARRAY,COUNT,RESULT
LOCAL ADDITION MOV ESI, 0 MOV ECX, COUNT
ADDITION: MOV EAX, ARRAY[ESI*4] ;;双字为4字节
X
MOV TERMINAL, 0
ADD RESULT, EAX ADC RESULT+4, 0 INC ESI LOOP ADDITION ENDM
;;将进位加到结果的高位双字中
7.17 为下列数据段中的数组编制一程序,调用题7.16的宏定义SUMMING,求出该数组中各元素之和。
DATA DD 101246,274365,843250,475536 SUM DQ ? 答:程序如下:
SUMMING MACRO ARRAY,COUNT,RESULT
LOCAL ADDITION MOV ESI, 0 MOV ECX, COUNT
ADDITION: MOV EAX, ARRAY[ESI*4] ;;双字为4字节
ADD RESULT, EAX ADC RESULT+4, 0 ;;将进位加到结果的高位双字中 INC ESI LOOP ADDITION ENDM
.MODEL SMALL .386 .DATA
DATA DD 101246,274365,843250,475536 SUM DQ ?
.CODE
START: MOV AX, @DATA
MOV DS, AX
SUMMING DATA, 4, SUM MOV AX, 4C00H INT 21H END START 7.18 如把题7.16中的宏定义存放在一个宏库中,则题7.17的程序应如何修改?
答:程序修改如下:
INCLUDE MACRO.MAC ;假设存放的宏库名为MACRO.MAC .MODEL SMALL .386 .DATA
DATA DD 101246,274365,843250,475536 SUM DQ ?
.CODE
START: MOV AX, @DATA
MOV DS, AX
SUMMING DATA, 4, SUM MOV AX, 4C00H INT 21H END START
第 八 章. 习 题
8.1 写出分配给下列中断类型号在中断向量表中的物理地址。
(1) INT 12H (2) INT 8
答:(1) 中断类型号12H在中断向量表中的物理地址为00048H、00049H、0004AH、0004BH;
(2) 中断类型号8在中断向量表中的物理地址为00020H、00021H、00022H、00023H。 8.2 用CALL指令来模拟实现INT 21H显示字符T的功能。
答:MOV AH, 2
MOV DL, ‘T’ PUSH DS PUSHF ;因中断服务程序的返回指令是IRET,而不是RET MOV BX, 0 MOV DS, BX CALL DWORD PTR[21H*4] ;用CALL指令调用21H的中断服务程序 POP DS 8.3 写出指令将一个字节数据输出到端口25H。 答:指令为:OUT 25H, AL 8.4 写出指令将一个字数据从端口1000H输入。 答:指令为: MOV DX, 1000H
IN AX, DX
8.5 假定串行通讯口的输入数据寄存器的端口地址为50H,状态寄存器的端口地址为51H,状态寄存器
各位为1时含义如右图所示,请编写一程序:输入一串字符并存入缓冲区BUFF,同时检验输入的正确性,如有错则转出错处理程序ERROR_OUT。 答:程序段如下:
7 6 5 4 3 2 1 0 MOV DI, 0
MOV CX, 80 ;最多输入80个字符
格溢奇输输BEGIN: IN AL, 51H ;查询输入是否准备好?
式出偶入出TEST AL, 02H
错 错 校数寄JZ BEGIN
验据存IN AL, 50H ;输入数据并存入缓冲区BUFF
错 准器MOV BUFF[DI], AL
备空 INC DI
好 IN AL, 51H ;判断是否有错?
TEST AL, 00111000B 8.3 状态寄存器各位含义 JNZ ERROR_OUT LOOP BEGIN ┇ 8.6 试编写程序,它轮流测试两个设备的状态寄存器,只要一个状态寄存器的第0位为1,则就与其相
应的设备输入一个字符;如果其中任一状态寄存器的第3位为1,则整个输入过程结束。两个状态寄存器的端口地址分别是0024H和0036H,与其相应的数据输入寄存器的端口地址则为0026H和0038H,输入字符分别存入首地址为BUFF1和BUFF2的存储区中。 答:程序段如下:
MOV DI, 0 MOV SI, 0
BEGIN: IN AL, 24H
TEST AL, 08H ;查询第一个设备的输入是否结束? JNZ EXIT TEST AL, 01H ;查询第一个设备的输入是否准备好? JZ BEGIN1 IN AL, 26H ;输入数据并存入缓冲区BUFF1 MOV BUFF1[DI], AL INC DI
BEGIN1: IN AL, 36H
TEST AL, 08H ;查询第二个设备的输入是否结束
JNZ EXIT TEST AL, 01H JZ BEGIN IN AL, 38H MOV BUFF2[SI], AL INC SI JMP BEGIN ┇
;查询第二个设备的输入是否准备好?
;输入数据并存入缓冲区BUFF2
EXIT:
8.7 假定外部设备有一台硬币兑换器,其状态寄存器的端口地址为0006H,数据输入寄存器的端口地址
为0005H,数据输出寄存器的端口地址为0007H。试用查询方式编制一程序,该程序作空闲循环等待纸币输入,当状态寄存器第2位为1时,表示有纸币输入,此时可从数据输入寄存器输入的代码中测出纸币的品种,一角纸币的代码为01,二角纸币为02,五角纸币则为03。然后程序在等待状态寄存器的第3位变为1后,把应兑换的五分硬币数(用16进制表示)从数据输出寄存器输出。 答:程序段如下:
BEGIN: IN AL, 06H ;查询是否有纸币输入?
TEST AL, 04H JZ BEGIN IN AL, 05H ;测试纸币的品种 CMP AL, 01H ;是一角纸币吗? JNE NEXT1 MOV AH, 02 ;是一角纸币,输出2个5分硬币 JMP NEXT
NEXT1: CMP AL, 02H ;是二角纸币吗?
JNE NEXT2 MOV AH, 04 ;是二角纸币,输出4个5分硬币 JMP NEXT
NEXT2: CMP AL, 03H ;是五角纸币吗?
JNE BEGIN MOV AH, 10 ;是五角纸币,输出10个5分硬币
NEXT: IN AL, 06H ;查询是否允许输出5分硬币?
TEST AL, 08H JZ NEXT MOV AL, AH ;输出5分硬币 OUT 07H, AL JMP BEGIN 8.8 给定(SP)=0100H,(SS)=0300H,(FLAGS)=0240H,以下存储单元的内容为(00020)=0040H,
(00022)=0100H,在段地址为0900及偏移地址为00A0H的单元中有一条中断指令INT 8,试问执行INT 8指令后,SP,SS,IP,FLAGS的内容是什么?栈顶的三个字是什么?
答:执行INT 8指令后,(SP)=00FAH,(SS)=0300H,(CS)=0100H,(IP)=0040H,(FLAGS)=0040H
栈顶的三个字是:原(IP)=00A2H,原(CS)=0900H,原(FLAGS)=0240H 8.9 类型14H的中断向量在存储器的哪些单元里?
答:在0000:0050H,0000:0051H,0000:0052H,0000:0053H四个字节中。
8.10 假定中断类型9H的中断处理程序的首地址为INT_ROUT,试写出主程序中为建立这一中断向量而编
制的程序段。 答:程序段如下:
┇
MOV AL, 1CH ;取原中断向量,并保护起来 MOV AH, 35H INT 21H PUSH ES
PUSH BX PUSH DS
MOV AX, SEG INT_ROUT MOV DS, AX
MOV DX, OFFSET INT_ROUT MOV AL, 09H MOV AH, 25H ;设置中断向量功能调用 INT 21H POP DS ┇ POP DX ;还原原中断向量 POP DS MOV AL, 1CH MOV AH, 25H INT 21H
8.11 编写指令序列,使类型1CH的中断向量指向中断处理程序SHOW_CLOCK。
答:程序段如下:
┇
MOV AL, 1CH MOV AH, 35H ;取中断向量功能调用,取原中断向量 INT 21H PUSH ES PUSH BX PUSH DS
MOV AX, SEG SHOW_CLOCK MOV DS, AX
MOV DX, OFFSET SHOW_CLOCK MOV AL, 1CH MOV AH, 25H ;设置中断向量功能调用 INT 21H POP DS ┇ POP DX POP DS MOV AL, 1CH MOV AH, 25H ;设置中断向量功能调用,还原原中断向量 INT 21H ┇
8.12 如设备D1,D2,D3,D4,D5是按优先级次序排列的,设备D1的优先级最高。而中断请求的次序
如下所示,试给出各设备的中断处理程序的运行次序。假设所有的中断处理程序开始后就有STI指令。
(1) 设备D3和D4同时发出中断请求。
(2) 在设备D3的中断处理程序完成之前,设备D2发出中断请求。
(3) 在设备D4的中断处理程序未发出中断结束命令(EOI)之前,设备D5发出中断请求。 (4) 以上所有中断处理程序完成并返回主程序,设备D1,D3,D5同时发出中断请求。
答:各设备的中断处理程序的运行次序是:INT_D3,INT_D2嵌套INT_D3,INT_D4,INT_D5;
INT_D1,INT_D3,INT_D5。 8.13 在8.12题中假设所有的中断处理程序中都没有STI指令,而它们的IRET指令都可以由于FLAGS
出栈而使IF置1,则各设备的中断处理程序的运行次序应是怎样的?
答:各设备的中断处理程序的运行次序是:INT_D3,INT_D2,INT_D4,INT_D5;
INT_D1,INT_D3,INT_D5。
8.14 试编制一程序,要求测出任一程序的运行时间,并把结果打印出来。
答:程序段如下:
TITLE TEST_TIME.EXE ;测试程序运行时间程序 ;****************************************** DSEG SEGMENT ;定义数据段 COUNT DW 0 ;记录系统时钟(18.2次中断/秒)的中断次数 SEC DW 0 ;存放秒钟数 MIN DW 0 ;存放分钟数 HOURS DW 0 ;存放小时数
PRINTTIME DB 0DH, 0AH, ‘The time of exection program is:’ CHAR_NO EQU $- PRINTTIME DSEG ENDS ;以上定义数据段 ;****************************************** CSEG SEGMENT ;定义代码段 MAIN PROC FAR
ASSUME CS: CSEG, DS: DSEG
START: PUSH DS ;设置返回DOS
SUB AX, AX PUSH AX
MOV AX, DSEG MOV DS, AX ;给DS赋值
MOV AL, 1CH ;取原来的1CH中断向量 MOV AH, 35H INT 21H PUSH ES ;保存原来的1CH中断向量 PUSH BX
PUSH DS ;设置新的1CH中断向量 MOV AX, SEG CLINT MOV DS, AX
MOV DX, OFFSET CLINT MOV AL, 1CH MOV AH, 25H INT 21H POP DS
IN AL, 21H ;清除时间中断屏蔽位并开中断 AND AL, 0FEH OUT 21H, AL STI ┇ ;要求测试时间的程序段
POP DX ;恢复原来的1CH中断向量 POP DS MOV AL, 1CH MOV AH, 25H INT 21H
CALL PRINT ;打印输出测试时间 RET ;返回DOS
MAIN ENDP ;
----------------------------------------------------------------------------------
CLINT
PROC NEAR ;中断服务子程序 PUSH DS PUSH BX
MOV BX, SEG COUNT MOV DS, BX LEA BX, COUNT INC WORD PTR [BX] ;记录系统时钟的中断次数单元+1 CMP WORD PTR [BX],18 ;有1秒钟吗? JNE TIMEOK CALL INCTEST ;有1秒钟,转去修改时间 CMP HOURS, 12 ;有12小时吗? JLE TIMEOK SUB HOURS, 12 ;有12小时,将小时数减去12 MOV AL, 20H ;发中断结束命令 OUT 20H, AL POP BX POP DS IRET ENDP ;CLINT中断服务子程序结束
ADJ:
TIMEOK:
CLINT
;
----------------------------------------------------------------------------------
INCTEST PROC NEAR ;修改时间子程序
MOV WORD PTR [BX], 0 ;中断次数单元或秒单元或分单元清0 ADD BX, 2 INC WORD PTR [BX] ;秒单元或分单元或时单元+1 CMP WORD PTR [BX],60 ;有60秒或60分吗? JLE RETURN CALL INCTEST ;先修改秒单元,再修改分单元,再修改时单元
RETURN: RET INCTEST ENDP ;INCTEST子程序结束 ;
----------------------------------------------------------------------------------
PRINT PROC NEAR ;打印输出子程序
LEA BX, PRINTTIME ;打印输出PRINTTIME信息 MOV CX, CHAR_NO
ROTATE: MOV DL, [BX]
MOV AH, 05H INT 21H INC BX LOOP ROTATE
MOV BX, HOURS ;打印时间的小时数 CALL BINIDEC ;调二进制转换为10进制并打印输出子程序 MOV DL, ‘:’ ;打印输出冒号 ‘:’ MOV AH, 05H INT 21H MOV BX, MIN ;打印时间的分钟数 CALL BINIDEC MOV DL, ‘:’ MOV AH, 05H INT 21H MOV BX, SEC ;打印时间的秒钟数
CALL BINIDEC
PRINT ;PRINT子程序结束 ;
----------------------------------------------------------------------------------
BINIDEC PROC NEAR ;二进制转换为10进制子程序
MOV CX, 10000D CALL DEC _DIV ;调除法并打印输出子程序 MOV CX, 1000D CALL DEC _DIV MOV CX, 100D CALL DEC _DIV MOV CX, 10D CALL DEC _DIV MOV CX, 1D CALL DEC _DIV RET
BINIDEC ENDP ;BINIDEC子程序结束 ;
----------------------------------------------------------------------------------
DEC_DIV PROC NEAR ;除法并打印输出子程序
MOV AX, BX MOV DX, 0 DIV CX MOV BX, DX ;余数保存在(BX)中作下一次的除法 MOV DL, AL ;商(在00H~09H范围内)送(DL) ADD DL, 30H ;转换为0~9的ASCII码 MOV AH, 05H ;打印输出 INT 21H RET
DEC_DIV ENDP ;DEC_DIV子程序结束 ;
---------------------------------------------------------------------------------- CSEG ENDS ;以上定义代码段 ;******************************************
END START ;汇编语言源程序结束
RET ENDP
第 九 章. 习 题
9.1 INT 21H的键盘输入功能1和功能8有什么区别?
答:键盘输入功能1:输入字符并回显(回送显示器显示) (检测Ctrl_Break);
键盘输入功能8:输入字符但不回显(也检测Ctrl_Break)。
9.2 编写一个程序,接受从键盘输入的10个十进制数字,输入回车符则停止输入,然后将这些数字加
密后(用XLAT指令变换)存入内存缓冲区BUFFER。加密表为:
输入数字:0,1,2,3,4,5,6,7,8,9 密码数字:7,5,9,1,3,6,8,0,2,4
答:程序段如下:
SCODE DB 7, 5, 9, 1, 3, 6, 8, 0, 2, 4 ;密码数字 BUFFER DB 10 DUP (?) ; ┇
MOV SI, 0 MOV CX, 10 LEA BX, SCODE
INPUT: MOV AH, 1
INT 21H CMP AL, 0DH JZ EXIT SUB AL, 30H JB INPUT CMP AL, 09H JA INPUT XLAT MOV BUFFER[SI], AL INC SI LOOP INPUT
EXIT: RET
;从键盘输入一个字符的功能调用
;输入回车符则停止输入 ;是0~9吗?
;换为密码 ;保存密码
9.3 对应黑白显示器屏幕上40列最下边一个象素的存储单元地址是什么?
答:对应黑白显示器屏幕上40列最下边一个象素的存储单元地址是:B000:0F78H 9.4 写出把光标置在第12行,第8列的指令。
答:指令如下:
MOV DH, 0BH ;0BH=12-1 MOV DL, 07H ;07H=8-1 MOV BH, 0 MOV AH, 2 ;置光标功能调用 INT 10H 9.5 编写指令把12行0列到22行79列的屏幕清除。
答:指令如下:
MOV AL, 0 ;清除屏幕 MOV BH, 07 MOV CH, 12 ;左上角行号 MOV CL, 0 ;左上角列号 MOV DH, 22 ;右下角行号 MOV DL, 79 ;右下角列号 MOV AH, 6 ;屏幕上滚功能调用 INT 10H
9.6 编写指令使其完成下列要求。
(1) 读当前光标位置
(2) 把光标移至屏底一行的开始
(3) 在屏幕的左上角以正常属性显示一个字母M 答:指令序列如下:
(1) MOV AH, 3 ;读当前光标位置,返回DH/DL=光标所在的行/列
MOV BH, 0 INT 10H
(2) MOV DH, 24 ;设置光标位置
MOV DL, 0 MOV BH, 0 MOV AH, 2 INT 10H
(3) MOV AH, 2 ;设置光标位置
MOV DX, 0 MOV BH, 0 INT 10H MOV AH, 9 ;在当前光标位置显示一个字符 MOV AL, ‘M’
MOV BH, 0 MOV BL, 7 MOV CX, 1 INT 10H
9.7 写一段程序,显示如下格式的信息:
Try again, you have n starfighters left.
其中n为CX寄存器中的1~9之间的二进制数。 答:程序段如下:
MESSAGE DB ‘Try again, you have ’ CONT DB n
DB ‘ starfighters left.$’
; ┇
ADD CL, 30H MOV CONT, CL ;保存ASCII码 LEA DX, MESSAGE MOV AH, 9 ;显示一个字符串的DOS调用 INT 21H
9.8 从键盘上输入一行字符,如果这行字符比前一次输入的一行字符长度长,则保存该行字符,然后
继续输入另一行字符;如果它比前一次输入的行短,则不保存这行字符。按下‘$’输入结束,最后将最长的一行字符显示出来。 答:程序段如下:
STRING DB 0 ;存放字符的个数
DB 80 DUP (?), 0DH,0AH,‘$’ ;存放前一次输入的字符串,兼作显示缓冲区
BUFFER DB 80 ;输入字符串的缓冲区,最多输入80个字符
DB ?
DB 80 DUP (20H)
; ┇ INPUT: LEA DX, BUFFER ;输入字符串
MOV AH, 0AH ;输入字符串的DOS调用 INT 21H LEA SI, BUFFER+1 ;比较字符串长度 LES DI, STRING MOV AL, [SI] CMP AL, [DI] JBE NEXT MOV CX, 80+1 ;大于前次输入的字符串,更换前次的字符串 CLD REP MOVSB
NEXT: MOV AH, 1 ;输入结束符吗?
INT 21H CMP AL, ‘$’ ;是结束符吗? JNE INPUT ;不是则继续输入 LEA DX, STRING+1 ;显示字符串 MOV AH, 9 ;显示一个字符串的DOS调用 INT 21H 9.9 编写程序,让屏幕上显示出信息“What is the date (mm/dd/yy)?”并响铃(响铃符为07),然后
从键盘接收数据,并按要求的格式保存在date存储区中。 答:程序段如下:
MESSAGE DB ‘What is the date (mm/dd/yy)?’, 07H, ‘$’ DATAFLD DB 10, 0 DATE DB 10 DUP (‘ ’)
;
┇
MOV AH, 9 LEA DX, MESSAGE INT 21H MOV AH, 0AH LEA DX, DATAFLD INT 21H
;显示一个字符串的DOS调用
;显示字符串
;输入字符串的DOS调用
9.10 用户从键盘输入一文件并在屏幕上回显出来。每输入一行(≤80字符),用户检查一遍, 如果用户
认为无需修改,则键入回车键,此时这行字符存入BUFFER缓冲区保存,同时打印机把这行字符打印出来并回车换行。 答:程序段如下:
INAREA DB 80 ;输入字符串的缓冲区,最多输入80个字符 ACTLEN DB ?
BUFFER DB 80 DUP (?) ; ┇ INPUT: LEA DX, INAREA ;输入字符串
MOV AH, 0AH ;输入字符串的DOS调用 INT 21H CMP ACTLEN, 0 JE EXIT MOV BX, 0 MOV CH, 0
MOV CL, ACTLEN
PRINT: MOV AH, 5 ;打印输出
MOV DL, BUFFER[BX] INT 21H INC BX LOOP PRINT MOV AH, 5 ;打印输出回车换行 MOV DL, 0AH INT 21H MOV DL, 0DH INT 21H JMP INPUT
EXIT: RET 9.11 使用MODE命令,设置COM2端口的通信数据格式为:每字8位,无校验,1位终止位,波特率为
1200b/s。
答:命令格式如下:
MODE COM2:12,N,8,1
第 十 章. 习 题
10.1 写出指令,选择显示方式10H,并将背景设为绿色。
答: MOV AH, 00H
MOV AL, 10H ;选择显示方式10H(16色图形) INT 10H MOV AH, 10H MOV AL, 00H
MOV BH, 10H MOV BL, 0 INT 10H
设置背景色也可用:
MOV AH, 0BH MOV BH, 0 MOV BL, 8 INT 10H
;背景设为绿色(02H也可以,是用DEBUG调试出来的) ;选择0号调色板
;设置背景色和调色板 ;设置背景色功能 ;绿色背景
10.2 如何使用INT 10H的功能调用改变显示方式?
答:在AH中设置功能号00H,在AL中设置显示方式值,调用INT 10H即可。 10.3 VGA独有的一种显示方式是什么?
答:像素值为640×480,可同时显示16种颜色,这种显示方式(12H)是VGA独有的。 10.4 对于EGA和VGA显示适配器,使用显示方式13H时(只有VGA有),显示数据存在哪里?
答:显示数据存在显示存储器里。 10.5 对于VGA的显示方式13H时存放一屏信息需要多少字节的显存?
答:需要64000个字节。
10.6 利用BIOS功能编写图形程序:设置图形方式10H,选择背景色为蓝色,然后每行(水平方向)显示
一种颜色,每4行重复一次,一直到整个屏幕都显示出彩条。 答:程序如下:
TITLE GRAPHIX.COM codeseg segment
assume cs:codeseg, ds:codeseg, ss:codeseg org 100h
main proc far
mov ah, 00h mov al, 10h ;选择显示方式10h(16色图形) int 10h mov ah, 0bh mov bh, 00h mov bl, 01h ;背景设为蓝色 int 10h mov ah, 0bh mov bh, 01h mov bl, 00h ;设置调色板0# int 10h
mov bx, 0 ;显存的第0页 mov cx, 0 ;起始列号为0列 mov dx, 0 ;起始行号为0行
line: mov ah, 0ch ;写像素点
mov al, bl int 10h inc cx
cmp cx, 640 jne line
mov cx, 0 ;起始列号为0列 inc bl
and bl, 03h ;只显示四种颜色(因此保留最低两位) inc dx
cmp dx, 350 jne line int 20h
main endp codeseg ends
end main
10.7 修改10.6题的程序,使整个屏幕都显示出纵向的彩条。
答:程序如下:
TITLE GRAPHIX.COM codeseg segment
assume cs:codeseg, ds:codeseg, ss:codeseg org 100h
main proc far
mov ah, 00h mov al, 10h ;选择显示方式10h(16色图形) int 10h mov ah, 0bh mov bh, 00h mov bl, 01h ;背景设为蓝色 int 10h mov ah, 0bh mov bh, 01h mov bl, 00h ;设置调色板0# int 10h
mov bx, 0 ;显存的第0页 mov cx, 0 ;起始列号为0列 mov dx, 0 ;起始行号为0行
line: mov ah, 0ch ;写像素点
mov al, bl int 10h inc dx
cmp dx, 350 jne line
mov dx, 0 ;起始行号为0行 inc bl
and bl, 03h ;只显示四种颜色(因此保留最低两位) inc cx
cmp cx, 640 jne line int 20h
main endp codeseg ends
end main
10.8 按动键盘上的光标控制键,在屏幕上下左右任一方向上绘图,每画一点之前,由数字键0~3指定
该点的颜色值,按动ESC键,绘图结束,返回DOS。 答:程序如下:
;DRAW—Program to draw on screen with sursor arrows ;For 640*350 color mode up equ 48h ;向上键的扫描值 down equ 50h ;向下键的扫描值 left equ 4bh ;向左键的扫描值 right equ 4dh ;向右键的扫描值 escape equ 1bh ;“Esc” character codeseg segment main proc far
assume cs:codeseg
;clear screen by scrolling it, using ROM call start: mov ah, 06h
mov al, 00h mov cx, 00h mov dl, 79 mov dh, 24 int 10h
;screen pointer will be in CX,DX registers;row number (0 to 350d) in DX ;coumn number (0 to 640d) in CX
mov ah, 00h mov al, 10h ;选择显示方式10h(16色图形) int 10h mov ah, 0bh mov bh, 00h mov bl, 01h ;背景设为蓝色 int 10h mov ah, 0bh mov bh, 01h mov bl, 00h ;设置调色板0# int 10h
mov dx, 175 ;设在屏幕中心 mov cx, 320
;get character from keyboard get_char: mov ah, 0 ;键盘输入
int 16h
cmp al, escape jz exit
cmp al, 33h ;>‘3’吗? jg plot
cmp al,30h ;<‘0’吗? jl plot
mov bl, al ;是‘0’~‘3’,设置颜色 and bl, 03 jmp get_char
;figure out which way to go, and draw new line plot: mov al, ah
cmp al, up jnz not_up dec dx
not_up: cmp al, down
jnz not_down inc dx
not_down: cmp al, right
jnz not_right inc cx
not_ right: cmp al, left
jnz write dec cx
;use ROM routine to write dot,reguires row# in DX,col in CX,color in AL write: mov al, bl
mov ah, 0ch int 10h
jmp get_char
exit: int 20h main endp codeseg ends
end start
10.9 位屏蔽寄存器的作用是什么?在16色,640×480显示方式中如何使用位屏蔽寄存器?
答:位屏蔽寄存器的作用是决定了新的像素值产生的方法。当位屏蔽寄存器的某位设为0时,相
对应的像素值直接由锁存器写入显存;位屏蔽寄存器的某位为1时,所对应的像素值由锁存器中的像素值与CPU数据或置位/重置寄存器中相应位合并之后产生。 10.10 读映像选择寄存器的作用是什么?如果4个位面的内容都需要读取,读映像选择寄存器应如何设
置?
答:读映像选择寄存器的作用是用于选择哪一个位面的字节读入CPU。读映像选择寄存器的0和
1位,用来指定哪个位面的锁存器内容读到CPU。如果4个位面的内容都需要读取,则必须对同一地址执行4次读操作,在每次读之前,用指令分别设置读映像选择寄存器。 10.11 编写程序使一只“鸟”飞过屏幕。飞鸟的动作可由小写字母v (ASCII码76H)变为破折号(ASCII
码0C4H)来模仿,这个字符先后交替在两列显示。鸟的开始位置是0列20行,每个字符显示0.5秒,然后消失。 答:程序段如下:
TITLE Flier.EXE ;飞鸟程序 ;****************************************** DSEG SEGMENT ;定义数据段 BIRD DB 76H, 07 ;小写字母v及属性
DB 0C4H, 07 ;破折号及属性
DSEG ENDS ;以上定义数据段 ;****************************************** CSEG SEGMENT ;定义代码段 MAIN PROC FAR
ASSUME CS: CSEG, DS: DSEG
START: PUSH DS ;设置返回DOS
SUB AX, AX PUSH AX
MOV AX, DSEG MOV DS, AX ;给DS赋值
MOV AH, 0FH ;取当前显示方式 INT 10H PUSH AX ;保存当前显示方式(AL) MOV AH, 0 ;设置彩色80×25文本方式 MOV AL, 3 INT 10H MOV DH, 20 ;20行 MOV DL, 0 ;0列
BEGIN: MOV SI, 2 ;字符v和破折号“-”交替显示
MOV CX, 1 ;一次显示一个字符及属性 LEA DI, BIRD
DISP: CMP DL, 79 ;飞到79列就退出
JAE EXIT MOV AH, 2 ;置光标位置 INT 10H MOV AH, 9 ;在光标位置显示字符及属性 MOV AL, [DI] ;取显示字符及属性 MOV BL, [DI+1]
INT 10H CALL DELAY MOV AH, 9 MOV AL, ‘ ’ MOV BL, 7 INT 10H INC DL ADD DI, 2 DEC SI JNZ DISP JMP BEGIN POP AX MOV AH, 0 INT 10H RET ENDP
;延时0.5秒
;在光标位置显示字符及属性 ;显示空格,擦除该位置的字符
;飞到下一列
EXIT: ;恢复当前显示方式(AL)
;返回DOS
MAIN
;
----------------------------------------------------------------------------------
DELAY PROC NEAR ;延时0.5s子程序
PUSH CX PUSH DX
MOV DX, 50 ;延时0.5s
DEL1: MOV CX, 2801 ;延时10ms DEL2: LOOP DEL2
DEC DX JNZ DEL1 POP DX POP CX RET
DELAY ENDP ;DELAY子程序结束 ;
---------------------------------------------------------------------------------- CSEG ENDS ;以上定义代码段 ;******************************************
END START ;汇编语言源程序结束
10.12 用图形文本的方法设计“Name=XXX”(X为你自己姓名的缩写),并将其数据编码定义在一个数组
中。
ASCII码为0DCH 答:用图形文本的方法设计“NAME=YQS”的程序和数组如下:
显示格式如下:
ASCII码为0DFH S h o o t i n g
TITLE NAME_YQS.EXE ;显示“NAME=YQS”的程序 ;******************************************
;Graphics block message for the words shooting NAME=YQS
ASCII码为0DDH ASCII码为0DEH ASCII码为0DBH
;00H→end of massage,0FFH→end of screen line DSEG SEGMENT ;定义数据段
DB 2 ;Start row(开始行) DB 2 ;Start column(开始列) DB 1000 0011B ;Color attribute
DB ‘Shooting’,0FFH,0FFH ;显示“Shooting” DB 7 DUP(0DCH),0FFH,0FFH
;Graphics encoding of the word NAME=YQS using IBM character set
DB 0DEH, 0DBH, 4 DUP(20H), 0DBH, 0DDH, 20H, 0DBH, 0DFH, 0DBH DB 20H, 20H, 0DBH, 5 DUP(20H), 0DBH, 20H, 2 DUP(0DFH, 0DBH) DB 8 DUP(20H), 0DFH, 0DBH, 20H, 20H, 0DBH, 0DFH, 20H, 20H DB 3 DUP(0DBH), 3 DUP(20H), 3 DUP(0DBH), 0DCH, 0FFH
DB 0DEH, 0DBH, 0DBH, 3 DUP(20H), 0DBH, 0DDH, 2 DUP(20H, 0DBH) DB 20H, 20H, 0DBH, 0DBH, 3 DUP(20H), 0DBH, 0DBH, 20H, 20H, 0DBH DB 11 DUP(20H), 3 DUP(0DBH, 20H, 20H), 20H, 0DBH, 20H, 0DBH DB 3 DUP(20H), 0DFH, 0FFH
DB 0DEH, 0DBH, 20H, 0DBH, 20H, 20H, 0DBH, 0DDH, 2 DUP(20H, 0DBH) DB 20H, 4 DUP(20H, 0DBH), 20H, 20H, 0DBH, 0DCH, 0DBH, 20H
DB 7 DUP(0DFH), 3 DUP(20H, 20H, 0DBH), 3 DUP(20H), 0DBH, 20H, 0DBH DB 3 DUP(0DCH), 20H, 0FFH
DB 0DEH, 0DBH, 20H, 20H, 0DBH, 20H, 0DBH, 0DDH, 20H, 0DBH, 0DFH DB 0DBH, 4 DUP(20H, 20H, 0DBH), 20H, 0DFH, 20H, 7 DUP(0DCH), 20H DB 20H, 0DFH, 0DBH, 0DBH, 0DFH, 20H, 20H, 0DBH, 3 DUP(20H), 0DBH DB 20H, 20H, 3 DUP(0DFH), 0DBH, 0FFH
DB 0DEH, 0DBH, 3 DUP(20H), 0DBH, 0DBH, 0DDH, 2 DUP(20H, 0DBH), 20H DB 20H, 0DBH, 5 DUP(20H), 2 DUP(0DBH, 20H, 20H), 10 DUP(20H), 0DBH DB 0DBH, 3 DUP(20H), 0DBH, 20H, 0DCH, 20H, 0DBH, 20H, 0DCH DB 3 DUP(20H), 0DBH, 0FFH
DB 0DEH, 0DBH, 4 DUP(20H), 0DBH, 0DDH, 0DCH, 0DBH, 20H, 0DBH DB 0DCH, 20H, 0DBH, 5 DUP(20H), 0DBH, 20H, 2 DUP(0DCH, 0DBH) DB 9 DUP(20H), 0DCH, 0DBH, 0DBH, 0DCH, 3 DUP(20H), 0DFH, 0DFH DB 0DBH, 20H, 20H, 0DFH, 3 DUP(0DBH), 20H, 0FFH DB 00 ;结束显示标志
START_COL DB ? DSEG ENDS ;以上定义数据段 ;******************************************
;Text display procedures: display a message on the graphics screen CSEG SEGMENT ;定义代码段 MAIN PROC FAR
ASSUME CS: CSEG, DS: DSEG
START: PUSH DS ;设置返回DOS
SUB AX, AX PUSH AX
MOV AX, DSEG MOV DS, AX
LEA DI, NAME_YQS MOV DH, [DI] ;Get row into DH INC DI ;Bump pointer MOV DL, [DI] ;And column into DL MOV START_COL, DL ;Store start columnNAME_YQS
MOV AH, 2 ;Set cursor position MOV BH, 0 ;Page 0 INT 10H INC DI ;Bump pointer to attribute MOV BL, [DI] ;Get color code into BL
Char_write: INC DI ;Bump to message start
MOV AL, [DI] ;Get character CMP AL, 0FFH ;End of line? JE BUMP_ROW ;Next row CMP AL, 0 ;Test for terminator JE END_TEXT ;Exit routine CALL SHOW_CHAR JMP CHAR_WRITE
END_TEXT: RET ;返回DOS Bump_row: INC DH ;Row control register
MOV DL, START_COL ;Column control to start column MOV AH, 2 ;Set cursor position MOV BH, 0 ;Page 0 INT 10H JMP CHAR_WRITE
MAIN ENDP ;
----------------------------------------------------------------------------------
;Display character in AL and using the color code in BL Show_char PROC NEAR ;显示字符子程序
MOV AH, 9 ;BIOS service request number MOV BH, 0 ;Page 0 MOV CX, 1 ;No repeat INT 10H
;Bump cursor
INC DL MOV AH, 2 ;Set cursor position MOV BH, 0 ;Page 0 INT 10H RET
Show_char ENDP ;SHOW_CHAR子程序结束 ;
---------------------------------------------------------------------------------- CSEG ENDS ;以上定义代码段 ;******************************************
END START ;汇编语言源程序结束
10.13 游戏程序常常用随机数来控制其图形在屏幕上移动。请编写一程序,用随机数来控制笑脸符
(ASCII码02H)显示的位置。笑脸符每次显示的列号总是递增1。而行的位置可能是前次的上一行,下一行或同一行,这根据随机数是0、1或2来决定,当行号变为0、24或列号变为79时显示结束。笑脸在每个位置上显示0.25s。(提示:INT 1AH的AH=0是读当前时间的功能调用,利用该功能返回的随时都在变化的时间值作为产生随机数的基数。) 答:程序段如下:
TITLE Disp_Laugh.EXE ;笑脸显示程序 ;****************************************** CSEG SEGMENT ;定义代码段
MAIN START:
PROC FAR
ASSUME CS: CSEG PUSH DS
;设置返回DOS
SUB AX, AX PUSH AX
MOV AH, 0FH INT 10H PUSH AX MOV AH, 0 MOV AL, 3 INT 10H
MOV CX, 1 MOV DH, 12H MOV DL, 0
BEGIN:
CMP DL, 79 JAE EXIT CMP DH, 0 JBE EXIT CMP DH, 24 JAE EXIT MOV AH, 2 INT 10H MOV AH, 9 MOV AL, 02H MOV BL, 7 INT 10H CALL DELAY MOV AH, 9 MOV AL, ‘ ’ MOV BL, 7 INT 10H INC DL PUSH DX
MOV AH, 0 INT 1AH MOV AX, DX POP DX AND AL, 03H JZ DOWN CMP AL, 1 JNZ LEVEL DEC DH JMP BEGIN INC DH JMP BEGIN POP AX MOV AH, 0 INT 10H RET ENDP
;取当前显示方式
;保存当前显示方式(AL) ;设置彩色80×25文本方式
;一次显示一个笑脸字符及属性 ;12行,从屏幕左边的中间开始 ;0列
;移到79列就退出 ;移到第0行就退出 ;移到第24行就退出 ;置光标位置
;在光标位置显示字符及属性 ;取笑字符及属性
;延时0.25秒
;在光标位置显示字符及属性 ;显示空格,擦除该位置的字符
;移到下一列
;读当前时间, CH:CL=时:分,DH:DL=秒:1/100秒
;产生随机数基数
;随机数为1/100秒的最低两位 ;随机数的最低两位为0则下降一行
;随机数的最低两位为≥2则水平移动 ;随机数的最低位为1则上跳一行
DOWN: LEVEL: EXIT:
;恢复当前显示方式(AL)
;返回DOS
MAIN ;
----------------------------------------------------------------------------------
DELAY
PROC NEAR
;延时0.25s子程序
PUSH CX PUSH DX
MOV DX, 25 MOV CX, 2801 LOOP DEL2 DEC DX JNZ DEL1 POP DX POP CX RET ENDP
;延时0.25s ;延时10ms
DEL1: DEL2:
DELAY ;DELAY子程序结束 ;
---------------------------------------------------------------------------------- CSEG ENDS ;以上定义代码段 ;******************************************
END START ;汇编语言源程序结束
10.14 分配给PC机主板上的8253/54定时器的端口地址是什么?
答:8253/54定时器的3个独立计数器Counter0、Counter1和Counter2的端口地址分别为40H、
41H和42H。8253/54内部还有一个公用的控制寄存器,端口地址为43H。 10.15 8253/54定时器的三个计数器,哪一个用于扬声器?它的端口地址是什么?
答:8253/54定时器的计数器Counter2用于扬声器,它的端口地址为42H。
10.16 下面的代码是利用监控端口61H的PB4来产生延迟时间的,它适用于所有的286、386、Pentium PC
及兼容机。请指出该程序的延迟时间是多少?
MOV DL, 200
BACK: MOV CX, 16572 WAIT: IN AL, 61H
AND AL, 10H CMP AL, AH JE WAIT MOV AH, AL LOOP WAIT DEC DL JNZ BACK
答:该程序的延迟时间是200×16572×15.08μs=49981152μs≈50s。 10.17 在PC机上编写乐曲程序“Happy Birthday”,乐曲的音符及音频如下: 歌词 音符 音频 节拍 歌词 音符 音频 节拍 歌词 hap C 262 1/2 day C 262 1 so py C 262 1/2 to G 392 1 hap birtD 294 1 you F 349 2 py h day C 262 1 hap C 262 1/2 birth to F 349 1 py C 262 1/2 day you E 330 2 birtD 294 1 to h hap C 262 1/2 day A 440 1 you py C 262 1/2 dear F 349 1 birtD 294 1 so E 330 1 h 答:程序如下: 音符 音频 节拍 D 294 3 Bb 466 1/2 Bb 466 1/2 A C G F 440 262 392 349 1 1 1 2
TITLE
MUSIC — A music of ‘Happy Birthday’ ;连接时需加上GENSOUND程序
EXTRN SOUNDF: FAR ;SOUNDF是外部过程——通用发声程序 ;****************************************** STACK SEGMENT PARA STACK ‘STACK’ ;定义堆栈段
DB 64 DUP (‘STACK…’)
STACK ENDS ;以上定义堆栈段 ;****************************************** DSEG SEGMENT PARA ‘DATA’ ;定义数据段
MUS_FREQ DW 262, 262, 294, 262, 349, 330, 262, 262, 294, 262,392, 349, 262, 262
DW 294, 440, 349, 330, 294, 466, 466,440,262, 392, 349, -1
MUS_TIME DW 25, 25, 50, 50, 50, 100
DW 25, 25, 50, 50, 50, 100 DW 25, 25, 50, 50, 50, 50, 150 DW 25, 25, 50, 50, 50, 100
DSEG ENDS ;以上定义数据段 ;****************************************** CSEG SEGMENT PARA ‘CODE’ ;定义代码段
ASSUME CS: CSEG, DS: DSEG, SS: STACK
MUSIC PROC FAR
PUSH DS ;设置返回DOS SUB AX, AX PUSH AX
MOV AX, DSEG MOV DS, AX ;给DS赋值
LEA SI, MUS_FREQ ;取发声的频率(音阶)表首地址 LEA BP, MUS_TIME ;取发声的节拍(时间)表首地址
FREQ: MOV DI, [SI] ;读取频率值
CMP DI, -1 ;歌曲结束了吗? JE END_MUS MOV BX, DS:[BP] ;读取节拍 CALL SOUNDF ;调通用发声子程序 ADD SI, 2 ADD BP, 2 JMP FREQ
END_MUS: RET ;返回DOS MUSIC ENDP CSEG ENDS ;以上定义代码段 ;******************************************
END MUSIC ;汇编语言源程序结束
以下是SOUNDF——外部的通用发声子程序(教材392页) TITLE SOUNDF —— 通用发声子程序
;****************************************** PUBLIC SOUNDF ;定义为公共过程 ;****************************************** CSEG1 SEGMENT PARA ‘CODE’ ;定义代码段
ASSUME CS: CSEG1
SOUNDF PROC FAR
PUSH AX PUSH BX PUSH CX PUSH DX
PUSH DI
MOV AL, 0B6H ;写定时器8253的工作方式 OUT 43H, AL MOV DX, 12H ;根据频率求8253的计数值,即533H*896/freq MOV AX, 533H*896 ;(DX),(AX)=123280H=533H*896 DIV DI ;(DI) = freq OUT 42H, AL ;向8253送计数值 MOV AL, AH OUT 42H, AL IN AL, 61H ;取8255的PB口当前内容,并保护 MOV AH, AL OR AL, 3 ;开始发声,PB1=1,PB0=1 OUT 61H, AL
WAIT1: MOV CX, 663 ;延时(BX)×10ms
CALL WAITF MOV AL, AH AND AL, 0FCH ;停止发声,PB1=0,PB0=0 OUT 61H, AL POP DI POP DX POP CX POP BX POP AX RET
SOUNDF ENDP
;****************************************** WAITF PROC NEAR
PUSH AX
WAITF1: IN AL, 61H
AND AL, 10H CMP AL, AH JE WAITF1 MOV AH, AL LOOP WAITF1 POP AX RET
WAITF ENDP CSEG1 ENDS ;以上定义代码段 ;******************************************
END
10.18 编写用键盘选择计算机演奏歌曲的程序。首先在屏幕上显示出歌曲名单如下:
A MUSIC 1 B MUSIC 2 C MUSIC 3
当从键盘上输入歌曲序号A,B或C时,计算机则演奏所选择的歌曲,当在键盘上按下0键时,演奏结束。
答:程序段如下:
MUS_LST DB ‘A MUSIC 1’, 0DH, 0AH
DB ‘B MUSIC 2’, 0DH, 0AH DB ‘C MUSIC 3’, 0DH, 0AH DB ‘0 END’, 0DH, 0AH, ‘$’ ┇
MOV AH, 09
;显示字符串的DOS功能调用
INPUT:
LEA DX, MUS_LIST INT 21H MOV AH, 1 INT 21H
CMP AL, ‘0’ JE EXIT OR AL, 0010 0000B CMP AL, ‘a’ JNZ B0 CALL MUSIC1 JMP INPUT CMP AL, ‘b’ JNZ C0 CALL MUSIC2 JMP INPUT CMP AL, ‘c’ JNZ INPUT CALL MUSIC3 JMP INPUT RET
;键盘输入一个字符的DOS功能调用
;结束演奏吗? ;变为小写字母 ;演奏歌曲a吗?
;去演奏歌曲A
;演奏歌曲b吗?
B0:
;去演奏歌曲B
;演奏歌曲c吗?
C0:
;去演奏歌曲C
;返回
EXIT:
第 十一 章. 习 题
11.1 写出文件代号式磁盘存取操作的错误代码:
(1) 非法文件代号 (2) 路径未发现 (3) 写保护磁盘 答:错误代码为:
(1) 06 (2) 03 (4) 19 11.2 使用3CH功能建立一文件,而该文件已经存在,这时会发生什么情况? 答:此操作将文件长度置为0,写新文件,原文件内容被清除。 11.3 从缓冲区写信息到一个文件,如果没有关闭文件,可能会出现什么问题?
答:文件结尾的部分信息就没有被写入磁盘,从而造成写入的文件不完整。 11.4 下面的ASCIZ串有什么错误?
PATH_NAME DB ‘C:\\PROGRAMS\\TEST.DAT’
答:此ASCIZ串的最后少了一个全0字节,应改为:
PATH_NAME DB ‘C:\\PROGRAMS\\TEST.DAT’, 0 11.5 下面为保存文件代号定义的变量有什么错误?
FILE_HNDL DB ?
答:文件代号是字类型,因此应改为:
FILE_HNDL DW ?
11.6 在ASCPATH字节变量中为驱动器D的文件PATIENT.LST,请定义ASCIZ串。
答:ASCPATH DB ‘D:\\PATIENT.LST’, 0 11.7 对11.6题中的文件,它的每个记录包含:
病例号(patient number): 5字符, 姓名(name): 20字符, 城市(city): 20字符, 街道(street address): 20字符, 出生年月(mmddyy): 6字符, 性别(M/Fcode): 1字符, 病房号(room number): 2字符, 床号(bed number): 2字符, (1) 定义病人记录的各个域 (2) 定义保存文件代号的变量FHANDLE (3) 建文件 (4) 把PATNTOUT中的记录写入 (5) 关文件 (6) 以上文件操作包括测试错误
答:(1) PATNTOUT EQU THIS BYTE
patient DB 5 DUP (?) name DB 20 DUP (?) city DB 20 DUP (?) street DB 20 DUP (?) mmddyy DB 6 DUP (?) M_Fcode DB ? room DB 2 DUP (?) bed DB 2 DUP (?), 0AH, 0DH COUNT = $-PATNTOUT ;记录长度 (2) FHANDLE DW ? (3) MOV AH, 3CH ;建文件功能
MOV CX, 00 ;普通文件属性 LEA DX, ASCPATH INT 21H JC ERROR MOV FHANDLE, AX ;保存文件代号 (4) MOV AH, 40H ;写文件功能
MOV BX, FHANDLE ;取文件代号 MOV CX, COUNT ;记录长度 LEA DX, PATNTOUT ;记录的首地址 INT 21H JC ERROR CMP AX, COUNT ;所有的字节都写入了吗? JNE ERROR1 (5) MOV AH, 3EH ;关闭文件功能
MOV BX, FHANDLE ;取文件代号 INT 21H JC ERROR
(6) 文件操作的测试错误已包括在(3)、(4)、(5)的操作中。
11.8 对11.7题的文件,用文件代号式编写一个完整的读文件程序,读出的每个记录存入PATNTIN并在
屏幕上显示。 答:程序如下:
TITLE READDISP.EXE ;利用文件代号式顺序读文件程序 ;Read disk records created by hancreat
;-------------------------------------------------------------
.model small .stack 100h .data
endcde db 0 ;结束处理指示 fhandle dw ?
patntin db 80 DUP(‘ ’) ;DTA ascpath db ‘d:\\patient.lst’, 0
openmsg db ‘***open error***’, 0dh, 0ah readmsg db ‘***read error***’, 0dh, 0ah row db 0
;-------------------------------------------------------------
.code
begin proc far
mov ax, @data mov ds, ax mov es, ax
x86汇编语言程序设计教案及答案第
![](/skin/haowen/images/icon_star.png)
![](/skin/haowen/images/icon_star.png)
![](/skin/haowen/images/icon_star.png)
![](/skin/haowen/images/icon_star.png)