IBM-PC汇编语言程序设计(第二版)答案1-8章

更新时间:2023-11-26 10:01:01 阅读量: 教育文库 文档下载

说明:文章内容仅供预览,部分内容可能不全。下载后的文档,内容与下面显示的完全一致。下载之前请确认下面内容是否您想要的,是否完整无缺。

///第 一 章. (1) 369

习 题

(2) 10000

(3) 4095

(4) 32767

1.1 用降幂法和除法将下列十进制数转换为二进制数和十六进制数: 答:(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

(3) 1111 1111 1111 1111

(4) 1111 1111

答:(1) 10 1101b=2dh=45 (2) 1000 0000b=80h=128

(3) 1111 1111 1111 1111b=ffffh=65535 (4) 1111 1111b=ffh=255

1.3 将下列十六进制数转换为二进制数和十进制数: (1) fa

(2) 5b

(3) fffe

(4) 1234

答:(1) fah=1111 1010b=250 (2) 5bh=101 1011b=91

(3) fffeh=1111 1111 1111 1110b=65534 (4) 1234h=1 0010 0011 0100b=4660

1.4 完成下列十六进制数的运算,并转换为十进制数进行校核: (1) 3a+b7

(2) 1234+af

(3) abcd-fe

(4) 7ab×6f

答:(1) 3a+b7h=f1h=241 (2) 1234+afh=12e3h=4835 (3) abcd-feh=aacfh=43727 (4) 7ab×6fh=35325h=217893

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

(4) 59

答:(1) 4fh表示的十进制数为 79,4fh表示的字符为o; (2) 2bh表示的十进制数为 43,2bh表示的字符为 +; (3) 73h表示的十进制数为115,73h表示的字符为s;

(4) 59h表示的十进制数为89,59h表示的字符为y。 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。

2.6 在实模式下,存储器中每一段最多可有10000h个字节。如果用调试程序debug的r命令在终端上显示出当前各寄存器的内容如下,请画出此时存储器分段的示意图,以及条件标志of、sf、zf、cf的值。 c>debug -r

ax=0000 bx=0000 cx=0079 dx=0000 sp=ffee bp=0000 si=0000 di=0000 ds=10e4 es=10f4 ss=21f0 cs=31ff ip=0100 nv up di pl nz na po nc

答:此时存储器分段的示意图如右图所示。of、sf、zf、cf的值都为0。 2.7 下列操作可使用那些寄存器? (1) 加法和减法 (2) 循环计数

数据寄存器等

cx

(3) 乘法和除法 ax、dx,乘数和除数用其他寄存器或存储器 (4) 保存段地址

段寄存器 cs:ip ss:sp

zf=1

(5) 表示运算结果为0

(6) 将要执行的指令地址 答:答案见题目的右边。

(7) 将要从堆栈取出数据的地址

2.8 那些寄存器可以用来指示存储器地址?

答:bx、bp、si、di、堆栈操作时的sp、对应的段地址、386及其后继机型的exx。 2.9 请将下列左边的项和右边的解释联系起来(把所选字母放在括号中): (1) cpu (2) 存储器 (3) 堆栈 (4) ip (5) sp

(m)

(c) (d)

(a)

a.保存当前栈顶地址的寄存器。

b.指示下一条要执行的指令的地址。 d.以后进先出方式工作的存储空间。

e.把汇编语言程序翻译成机器语言程序的系统程序。 c.存储程序、数据等信息的记忆装置,微机有ram和rom两种。

(b)

(6) 状态标志 (l) (8) 段寄存器 (j) (9) 物理地址 (f) ss、es。

f.唯一代表存储空间中每个字节单元的地址。 g.能被计算机直接识别的语言。

h.用指令的助记符、符号地址、标号等符号书写程序的语言。 i.把若干个模块连接起来成为可执行文件的系统程序。

j.保存各逻辑段的起始地址的寄存器,8086/8088机有四个:cs、ds、 k.控制操作的标志,如df位。

l.记录指令操作结果的标志,共6位:of、sf、zf、af、pf、cf。 m.分析、控制并执行指令的部件,由算术逻辑部件alu和寄存器等 n.由汇编程序在汇编过程中执行的指令。

o.告诉cpu要执行的操作(一般还要指出操作数地址),在程序运行时

(7) 控制标志 (k)

(10) 汇编语言 (h) (11) 机器语言 (g) (12) 汇编程序 (e) (13) 连接程序 (i) 组成。 (14) 指令 (15) 伪指令 执行。

(o)

(n)

答:答案见题目的括号中。

第 三 章. 习 题

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 dx, [bx+(6–1)*2] (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 (2) mov ax, bx (3) mov ax, [1200h] (4) mov ax, [bx] (5) mov ax, 1100[bx] (6) mov ax, [bx][si] (7) mov ax, 1100[bx][si] 答:(1) (ax)=1200h (2) (ax)=0100h (3) (ax)=4c2ah (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指令的下一条指令的首地址。

改为:

mov bx, (6-1)*2

mov dx, block[bx]

也可

(2) jmp word ptr [bx] (3) jmp d[bx]

;(ip)=((ds)*10h+(bx))=0600h,pa=02600h

;(ip)=((ds)*10h+(bx)+d)=098ah,pa=0298ah

3.6 设当前数据段寄存器的内容为1b00h,在数据段的偏移地址2000h单元内,含有一个内容为0ff10h和8000h的指针,它们是一个16位变量的偏移地址和段地址,试写出把该变量装入ax的指令序列,并画图表示出来。

答:mov bx, [2000h] mov ax, [2000h+2] mov es, ax 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] (5) mov ax, [bx] (7) mov ax, [bp]

(4) mov ax, val

(6) mov ax, es:[bx] (8) mov ax, [si] (10) mov ax, val[bx]

操作数在本条指令中 操作数为 (bx)=0100h

(9) mov ax, [bx+10] (11) mov ax, [bx][si] 答:(1) 立即方式; (2) 寄存器寻址方式; (3) 直接寻址方式; (4) 直接寻址方式;

(12) mov ax, val[bx][si]

pa=20100h pa=20050h pa=20100h pa=15010h pa=200a0h pa=20110h pa=20150h pa=201a0h pa=201f0h

(5) bx寄存器间接寻址方式; (7) bp寄存器间接寻址方式; (8) si寄存器间接寻址方式; (9) bx寄存器相对寻址方式;

(6) 附加段bx寄存器间接寻址方式; pa=21100h

(10) bx寄存器相对寻址方式;

(11) bx和si寄存器基址变址寻址方式;

(12) bx和si寄存器相对基址变址寻址方式; array dw 23, 36, 2, 100, 32000, 54, 0 zero dw ?

3.9 在array数组中依次存储了七个字数据,紧接着是名为zero的字单元,表示如下:

(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寄存器的内容是什么? mov ax, table lea lea

ax, table

ax,table是将table单元的有效地址送到ax,(ax)=0032h

;000ah, 0014h, 001eh, 0028h, 0032h

答:mov ax, table是将table单元的内容送到ax,(ax)=1234h

3.11 执行下列指令后ax寄存器中的内容是什么? table dw 10, 20, 30, 40, 50 entry dw 3 ┇

mov bx, offset table add 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的内容变化过程示意图(标出存储单元的物理地址)。 答:mov ax,8057h; mov bx,0f79h; push ax; push bx; pop ax;

画图的话ss和sp内容已经知道,只要记住入栈的时候是先减后压,先高后低,出栈时候相反即可。 堆栈区和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) 答:(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 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 rcl

dx, 1

;((dx),(ax))*2 y, 6

mov ax, w

(2) z←w-(x+6)-(r+9) (4) z←((w-x)/5*y)*2

(3) z←(w*x)/(y+6),r←余数

;以下程序都未考虑带符号数的溢出

3.18 已知程序段如下:

mov ax, 1234h ;(ax)=1234h,标志位不变 mov cl, 4 rol

dec mul cx int

试问:

(1) 每条指令执行完后,ax寄存器的内容是什么? (2) 每条指令执行完后,进位、符号和零标志的值是什么? (3) 程序结束时,ax和dx的内容是什么? 答:(1) 见注释; (2) 见注释;

(3) (ax)=8d00h,(dx)=0

ax

;(ax)和标志位都不变

;(ax)=2341h,cf=1,sf和zf不变

;(ax)=2340h,cf=1不变,sf=0,zf=0

20h ax, cl

mov cx, 4 ;(ax)和标志位都不变

;(ax)=8d00h,cf=of=0,其它标志无定义

3.19 下列程序段中的每条指令执行完后,ax寄存器及cf、sf、zf和of的内容是什么? mov ax, 0 dec

ax

add ax, 7fffh add ax, 2 not ax sub sub sal

;(ax)=0,

;(ax)=7ffeh,

标志位不变

cf不变,sf=1,zf=0,of=0

cf=1,sf=0,zf=0,of=0

标志位不变

cf=1,sf=0,zf=1,of=1 cf=1,sf=1,zf=0,of=0

;(ax)=0ffffh,

;(ax)=8000h, cf=0,sf=1,zf=0,of=1

;(ax)=7fffh,

;(ax)=8000h, cf=1,sf=1,zf=0,of=1 ;(ax)=0ffffh,

ax, 0ffffh ax, 1 ax, 1 ax, 1

add ax, 8000h and ax, 58d1h sar neg ax

;(ax)=0,

;(ax)=58d1h, cf=0,sf=0,zf=0,of=0

;(ax)=0b1a2h, cf=0,sf=1,zf=0,of=1 ;(ax)=0d8d1h, cf=0,sf=1,zf=0,of=0 ;(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 div bl (7) mov

dx, datax+2

mov ax, datax div datay

3.21 写出对存放在dx和ax中的双字长数求补的指令序列。 答:neg dx neg ax sbb dx, 0

adc dx, 0

也可为:

not dx not ax

add ax, 1

ax, datax

mov bl, 23

3.22 试编写一程序求出双字长数的绝对值。双字长数在a和a+2单元中,结果存放在b和b+2单元中。 答:程序段如下: mov ax, a mov dx, a+2 cmp dx, 0 jns neg dx neg ax sbb

dx, 0

zhenshu: mov b, ax mov b+2, dx int

20h bx, value bx, value bx, 0ffh bx, 0 bx, 01h

;(bx)=9ah,cf、of都为0,af无定义,sf=1,zf=0,pf=1 ;(bx)=61h,cf、of都为0,af无定义,sf=0,zf=0,pf=0 ;(bx)=1ch,cf、of都为0,af无定义,sf=0,zf=0,pf=0 ;(bx)=00h,cf、of都为0,af无定义,sf=0,zf=1,pf=1

;(bx)=0e3h,cf、of都为0,af无定义,sf=1,zf=0,pf=0

3.23 假设(bx)=0e3h,变量value中存放的内容为79h,确定下列各条指令单独执行后的结果。 (1) xor (2) and (4) xor (5) and (6) test

zhenshu

;不是负数则转走

(3) or bx, value ;(bx)=0fbh,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 sal

al, 1 al, 1 al, 1 al, 1

;(al)=(+53*2)=6ah ;(al)=(-49*2)=9eh ;(al)=(53/2)= 1ah ;(al)=(-49/2)=0e7h

mov al, -49 mov al, 53 sar sar mov al, -49

3.26 试分析下面的程序段完成什么功能? mov cl, 04 shl shl shr or

dx, cl ax, cl bl, cl

dl, bl

mov bl, ah

答:本程序段将 ((dx),(ax)) 的双字同时左移4位,即将此双字乘以10h (16)。 3.27 假定(dx)=0b9h,(cl)=3,(cf)=1,确定下列各条指令单独执行后dx中的值。 (1) shr

dx, 1

;(dx)=05ch

(2) sar (3) shl (4) shl (5) ror (6) rol (7) sal (8) rcl (4) rcr

dx, cl dx, cl dl, 1 dx, cl dl, cl dh, 1 dx, cl dl, 1

;(dx)=17h ;(dx)=5c8h ;(dx)=72h ;(dx)=2017h ;(dx)=0cdh ;(dx)=0b9h ;(dx)=2cch ;(dx)=0dch

答:见注释。

3.28 下列程序段执行完后,bx寄存器的内容是什么? mov cl, 3 mov bx, 0b7h rol

bx,1

ror bx, cl 答:(bx)=0c02dh。

3.29 假设数据段定义如下: coname prline

db db

‘space explorers inc.’ 20 dup (‘’)

用串指令编写程序段分别完成以下功能: (1) 从左到右把coname中的字符串传送到prline。 (2) 从右到左把coname中的字符串传送到prline。 (3) 把coname中的第3和第4个字节装入ax。 (4) 把ax寄存器的内容存入从prline+5开始的字节中。

(5) 检查coname字符串中有无空格字符,如有则把第一个空格字符的地址传送给bx寄存器。 答:(1) mov cld mov mov mov lea lea rep (2) mov std mov mov mov lea add lea add rep (3) mov (4) mov

si, seg coname ds, si es, si si, coname si, 20-1 di, prline di, 20-1 movsb

ax, word ptr coname+3-1 word ptr prline +5, ax si, seg coname ds, si es, si si, coname di, prline movsb

cx, 20

cx, 20

(5) mov cld mov mov lea jne dec mov next:

al, ‘ ’ ;空格的ascii码送al寄存器

di, seg coname es, di di, coname next di bx, di ┇

repne scasb

3.30 编写程序段,把字符串string中的‘&’字符用空格符代替。 string db ‘the date is feb&03’ 答:程序段如下: mov cx, 18 mov al, ‘&’ cld

mov di, seg string mov es, di lea jne dec next:

di, string next di

;送空格符

db db db

30 dup (?) 9

dup (?) 132 dup (?)

repne scasb

mov es:byte ptr [di], ‘ ’ 3.31 假设数据段中数据定义如下: student_name student_addr print_line

分别编写下列程序段:

(1) 用空格符清除print_line域。 (2) 在student_addr中查找第一个‘-’。 (3) 在student_addr中查找最后一个‘-’。

(4) 如果student_name域中全是空格符时,填入‘*’。

(5) 把student_name移到print_line的前30个字节中,把student_ addr移到print_line的后9个字节中。 答:公共的程序段如下: mov mov (1) mov mov cld lea rep (2) mov mov

di, print_line stosb

cx, 9 al., ‘-’ di, ds es, di

cx, 132

;空格的ascii码送al寄存器

al., ‘ ’

cld lea jne dec (3) mov mov std lea add jne inc (4) mov mov cld lea repe jne mov mov lea rep (5) mov cld lea lea rep mov std lea lea rep

si, student_addr+9-1 di, print_line+132-1 movsb

si, student_name di, print_line movsb cx, 9

di, student_name scasb next cx, 30 al, ‘*’ stosb

cx, 30

;“*”的ascii码送al寄存器

di, student_name di, student_ addr di, 9-1 no_dash di

cx, 30

;空格的ascii码送al寄存器

di, student_ addr no_dash di

cx, 9

repne scasb

no_dash: ┇

al., ‘-’

repne scasb

no_dash: ┇

al, ‘ ’

next: ┇

3.32 编写一程序段:比较两个5字节的字符串olds和news,如果olds字符串不同于news字符串则执行new_less;否则顺序执行程序。 答:程序段如下: mov cx, 5 cld

mov di, seg olds mov ds, di mov es, di

lea lea jne ┇

si, olds di, news new_less

repe cmpsb

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 ja jg

exceed

bx, ax zero bx, ax bx, ax dx, cx exceed

dx, cx

(2) cmp (3) jcxz (4) cmp jo jle (5) cmp (6) cmp

overflow eq_sma

jbe eq_sma

3.34 试分析下列程序段: add ax, bx jno jnc sub jnc l3 jno jmp

l4 short l5 l1 l2 ax, bx

如果ax和bx的内容给定如下: ax bx (1) 147bh (2) b568h (3) 42c8h

80dch 42c8h 608dh

(4) d023h 9fd0h (5) 94b7h b568h

问该程序分别在上面5种情况下执行后,程序转向哪里? 答:(1) 转向l1 (2) 转向l1 (3) 转向l2

(4) 转向l5 (5) 转向l5

;因为加法指令后ax中已经是6ff3h ;因为加法指令后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 jg

l2 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位

fld5b fld6b fld7b

db ‘32654’ db 10 dup (0) db ‘part1’, 20

db ‘part2’, 50 db ‘part3’, 14 fld1wdw 0fff0h fld2wdw 01011001b fld3wdw fld7b fld4wdw 5, 6, 7, 8, 9 fld5wdw 5 dup (0) fld6wdw fld1w-fld1b datasg ends

4.8 假设程序中的数据定义如下: partno pname count dd plenth

dw db

?

? 16 dup (?)

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 address city

db 30 dup (?) db 15 dup (?) db 30 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 num data_list max min

segment

db 5

dw -1, 0, 2, 5, 4, 5 dup (?) dw ? dw ?

data_seg ends

;---------------------------------------------------------------- code_seg segment main

proc far

;设置返回dos

assume cs: code_seg, ds: data_seg start: push ds sub push ax

mov ax, data_seg mov ds, ax ; mov cx, 4 lea 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: jnle rout3 mov min, ax rout3: ret main

endp

code_seg ends

;---------------------------------------------------------------- end start

4.12 给出等值语句如下: alpha equ 100 beta equ 25 gamma

equ 2

;=2729h

下列表达式的值是多少? (1) alpha * 100 + beta

loop rout1

;程序段结束

cmp ax, min

;程序段开始 ;给ds赋值

ax, ax

bx, data_list

(2) alpha mod gamma + beta (3) (alpha +2) * beta – 2 (4) (beta / 3) mod 5 (6) alpha ge gamma (7) beta and 7 (8) gamma or 3 答:见注释。

;=19h ;=9f4h

;=3h

(5) (alpha +3) * (beta mod gamma) ;=67h

;=0ffffh

;=01h ;=03h

4.13 对于下面的数据定义,三条mov指令分别汇编成什么?(可用立即数方式表示) tablea tableb tablec ┇ mov mov mov

ax, length tablea bl, length tableb cl, length tablec

;汇编成mov ax, 000ah ;汇编成mov bl, 000ah ;汇编成mov cl, 0001h

dw 10 dup (?) db db

10 dup (?) ‘1234’

答:见注释。

4.14 对于下面的数据定义,各条mov指令单独执行后,有关寄存器的内容是什么? fldb db ? tablea tableb

dw 20 dup (?) db ‘abcd’

;(ax)=0001h ;(ax)=0002h ;(cx)=0014h ;(dx)=0028h

;(cx)=0001h

(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 答:见注释。

4.15 指出下列伪操作表达方式的错误,并改正之。 (1) data_seg (2) segment (3) mydata ┇ ends ┇

end main_proc main_proc endp 答:见注释。

4.16 按下面的要求写出程序的框架

(1) 数据段的位置从0e000h开始,数据段中定义一个100字节的数组,其类型属性既是字又是字节; (2) 堆栈段从小段开始,段组名为stack;

(3) 代码段中指定段寄存器,指定主程序从1000h开始,给有关段寄存器赋值; (4) 程序结束。 答:程序的框架如下: data_seg segment at 0e000h array_b array_w

label byte dw

50 dup (?)

;以上定义数据段

;main_proc

endp ;上下两句交换位置

end main_proc

;mydata ends(缺少段名字)

;删除end main_proc也可以

seg

;data_seg segment(伪操作错)

;segname segment ‘code’(缺少段名字)

‘code’ segment/data

;mydata segment

(4) main_proc proc far

data_seg ends

;---------------------------------------------------------------- stack_seg segment para stack ‘stack’ dw tos

100h dup (?) 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

;给ss赋值 ;给sp赋值

;设置返回dos

mov ss, ax push ds sub push ax

mov ax, data_seg mov ds, ax ┇ 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 augw augend sum d_seg e_seg addw addend e_seg c_seg main start: sub push ax mov ax, d_seg mov ds, ax mov ax, e_seg

;给ds赋值

segment label word dd dd

99251 ?

;以上定义数据段

;给ds赋值

;程序段部分

mov sp, offset tos

ax, ax

;----------------------------------------------------------------

ends segment

;----------------------------------------------------------------

label word dd

-15962

;以上定义附加段

ends segment proc far push ds ax, ax

;设置返回dos

;----------------------------------------------------------------

assume cs: c_seg, ds: d_seg, es: e_seg

mov es, ax ;

;给es赋值

;以下6条指令进行加法计算

mov ax, augw 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 c_seg 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

cl,byte ptr wval

;需要,如mov ;不需要

endp ends

;以上定义代码段

;----------------------------------------------------------------

(4) mov cl,wval (5) add al,bval+1 答:见注释。

第 五 章. 习 题

5.1 试编写一个汇编语言程序,要求对键盘输入的小写字母用大写字母显示出来。 答:程序段如下: begin: int jb ja

21h

;输入字符<‘a’吗? ;输入字符>‘z’吗?

;转换为大写字母,用and al, 1101 1111b也可

stop stop al, 20h

;显示一个字符的dos调用 mov ah, 1

;从键盘输入一个字符的dos调用

cmp al, ‘a’ cmp al, ‘z’ sub mov ah, 2 int

21h begin ret

jmp stop:

mov dl, al

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

答:程序段如下: begin: int jb ja

21h

;输入字符<‘a’吗? ;输入字符>‘z’吗?

;得到前导字符

stop stop al

;准备显示三个字符

;显示一个字符的dos调用

mov ah, 1

;从键盘输入一个字符的dos调用

cmp al, ‘a’ cmp al, ‘z’ dec mov cx, 3 display: int

inc stop:

mov dl, al

mov ah, 2 21h dl ret

loop display

5.3 将ax寄存器中的16位数分成4组,每组4位,然后把这四组数分别放在al、bl、cl和dl中。 答:程序段如下: dseg store dseg ┇ begin: lea a10:

mov cl, 4

;右移四次

segment db 4 dup (?) ends

mov ch, 4 ;循环四次

bx, store mov dx, ax

;取ax的低四位 ;低四位存入store中

;右移四次

;循环四次完了码?

;四组数分别放在al、bl、cl和dl中

and dx, 0fh mov [bx], dl inc shr dec jnz b10:

bx

ax, cl ch a10

mov dl, store

mov cl, store+1 mov bl, store+2 mov al, store+3 stop:

ret

5.4 试编写一程序,要求比较两个字符串string1和string2所含字符是否完全相同,若相同则显示‘match’, 若不相同则显示‘no match’。 答:程序如下: dseg string1 string2 yes no

segment

db ‘i am a student.’ db ‘i am a student!’ db ‘match’, 0dh, 0ah, ‘$’

db ‘no match’, 0dh, 0ah, ‘$’

dseg cseg main start: sub push ax

ends segment proc far push ds ax, ax

;设置返回dos

;--------------------------------------------------------------------------

assume cs: cseg, ds: dseg, es: dseg

mov ax, dseg mov ds, ax mov es, ax ; begin: lea cld

mov cx, string2 - string1 repe cmpsb jne lea jmp dispno: display: int ret main cseg end start

5.5 试编写一程序,要求能从键盘接收一个个位数n,然后响铃n次(响铃的ascii码为07)。 答:程序段如下: begin: int jb ja cbw mov cx, ax jcxz stop bell: int

mov dl, 07h

21h

;延时100ms

;准备响铃

mov ah, 2

;显示一个字符的dos调用,实际为响铃

;响铃次数n

sub

21h al, ‘0’ stop

stop

;输入字符<‘0’吗? ;输入字符>‘9’吗? mov ah, 1

;从键盘输入一个字符的dos调用

endp ends

;以上定义代码段

;串比较

;显示match

;显示no match

dispno dx, yes display lea 21h

dx, no

mov ah, 9

;显示一个字符串的dos调用

lea

si, string1

;设置串比较指令的初值

;给ds赋值 ;给es赋值

di, string2

;--------------------------------------------------------------------------

cmp al, 9

call delay100ms loop bell

stop: ret

5.6 编写程序,将一个包含有20个数据的数组m分成两个数组:正数数组p和负数数组n,并分别把这两个数组中数据的个数显示出来。 答:程序如下: dseg count array count1 array1 count2 array2 zhen fu

crlf dseg cseg main start: sub push ax mov ax, dseg mov ds, ax begin: lea lea lea begin1: js

mov cx, count bx, array si, array1 di, array2 mov ax, [bx]

;是负数码?

;是正数,存入正数数组

;正数个数+1

fushu count1

;给ds赋值

segment equ 20 dw 20 dup (?) db 0 db 0

dw 20 dup (?) dw 20 dup (?) ends segment proc far push ds ax, ax

;设置返回dos

;存放数组

;存放正数的个数 ;存放负数的个数

;正数的个数是:

;负数的个数是:

;存放正数 ;存放负数

db 0dh, 0ah, ‘the positive number is:’, ‘$’ db 0dh, 0ah, ‘$’

db 0dh, 0ah, ‘the negative number is:’, ‘$’

;--------------------------------------------------------------------------

assume cs: cseg, ds: dseg

cmp ax, 0 mov [si], ax inc add si, 2 jmp fushu: inc add di, 2 next: lea

short next mov [di], ax count2 add bx, 2 dx, zhen

;显示正数个数

;是负数,存入负数数组 ;负数个数+1

loop begin1 mov al, count1 call display

;调显示子程序

lea dx, fu

;显示负数个数

mov al, count2 call display ret main display int

endp proc near

21h

;将(al)中的二进制数转换为二个非压缩bcd码 ;变为0~9的ascii码

;显示子程序

;--------------------------------------------------------------------------

;调显示子程序

mov ah, 9 aam

;显示一个字符串的dos调用

add ah, ‘0’ mov dl, ah mov ah, 2 int

21h

;显示一个字符的dos调用

;变为0~9的ascii码

add al, ‘0’ mov dl, al mov ah, 2 int

21h dx, crlf

21h endp ends

lea int ret display cseg end start

;显示一个字符的dos调用

;显示回车换行

mov ah, 9 ;显示一个字符串的dos调用

;显示子程序结束 ;以上定义代码段

;--------------------------------------------------------------------------

5.7 试编写一个汇编语言程序,求出首地址为data的100d字数组中的最小偶数,并把它存放在ax中。 答:程序段如下: begin:

mov bx, 0

;取数组的第一个偶数

mov cx, 100

compare: mov ax, data[bx] add bx, 2 test ax, 01h jnz

;是偶数吗? ;没有偶数,退出

;取数组的下一个偶数

loopnz compare

stop

jcxz stop add bx, 2 test dx, 01h jnz jle

cmp ax, dx

next

;(ax)<(dx),则置换(ax)为最小偶数

loop compare1

mov ax, dx next:

;是偶数吗?

;不是,比较下一个数 ;(ax)<(dx)吗?

next

;不是,比较下一个数

;最后一个数是偶数,即为最小偶数,退出

compare1: mov dx, data[bx]

stop: ret

5.8 把ax中存放的16位二进制数k看作是8个二进制的“四分之一字节”。试编写程序要求数一下值为3(即11b)的四分之一字节数,并将该数(即11b的个数)在终端上显示出来。 答:程序段如下: begin: mov cx, 8

compare: test ax, 03h jnz inc noequal: ror ax, 1 loop compare add dl, ‘0’ mov ah, 2 int

21h ret

stop: 制数。

答:程序段如下: begin: mov ch, 4 mov cl, 4 input: int jb ja

21h

;<0吗?

;不是‘0~f’的数重新输入 ;是‘0~9’吗?

;不是,转‘a~f’的处理 ;转换为:0000b~1001b

input af

shl

bx, cl

;将前面输入的数左移4位

mov ah, 1 cmp al, 30h cmp al, 39h and al, 0fh jmp af: jb ja

;从键盘取数

mov bx, 0

;用于存放四位的16进制数

;将计数值转换为ascii码 ;进行显示

noequal dl

;是数03吗?

;不是,转走

;是,计数

;准备判断下一个数

mov dl, 0

;计数初始值

ror ax, 1

5.9 试编写一个汇编语言程序,要求从键盘接收一个四位的16进制数,并在终端上显示与它等值的二进

binary

and al, 1101 1111b ;转换为大写字母

;又

;不是‘a~f’的数重新输入 ;>f吗?

;不是‘a~f’的数重新输入 ;转换为:1010b~1111b

;将键盘输入的数进行组合

input input

cmp al, 41h cmp al, 46h and al, 0fh add al, 9 binary: del jnz dispn: disp: rol

or ch

bl, al

input bx, 1

mov cx, 16

;将16位二进制数一位位地转换成ascii码显示

mov dl, 0

rcl or int

dl, 1 dl, 30h

;进行显示

21h ret

mov ah, 2 loop disp stop:

5.10 设有一段英文,其字符变量名为eng,并以$字符结束。试编写一程序,查对单词sun在该文中的出现次数,并以格式“sun:xxxx”显示出次数。 答:程序如下: dseg eng disp dat dseg cseg main start: sub push ax mov ax, dseg mov ds, ax mov es, ax begin: lea comp: lea mov cx, 3 repe cmpsb jnz inc add bx, 2 nomatch: inc dec jnz done: mov cl, 4 lea done1:

bx, dat rol

;转换结果存入dat单元中

ax, cl

;取一位16进制数

dx comp mov ch, 4

;将次数转换为16进制数的ascii码

bx

;指向eng的下一个字母

ax

;串比较

;是,sun的个数加1

nomatch

;给ds赋值 ;给es赋值

;计算eng的长度(每次比较sun,因此比较次数-2)

segment db

‘here is sun, sun ,…,$’ db ‘sun:’

db ‘0000’ , 0dh, 0ah, ‘$’ ends segment proc far push ds ax, ax

;设置返回dos

keyword db ‘sun’

;--------------------------------------------------------------------------

assume cs: cseg, ds: dseg, es: dseg

mov ax, 0

mov dx, disp-eng-2

bx, eng

mov di, bx

si, keyword

mov dx, ax and dl, 0fh

add dl, 30h cmp dl, 39h jle

store

;是“a~f”所以要加7

;转换结果存入dat单元中

mov [bx], dl bx ch done1 lea 21h endp ends

;以上定义代码段

dx, disp

;显示字符串程序(将disp和dat一起显示)

add dl, 07h store: inc dec jnz display: int ret main cseg end start

5.11 从键盘输入一系列以$为结束符的字符串,然后对其中的非数字字符计数,并显示出计数结果。 答:程序段如下: dseg buff count dseg ┇ begin: input: int

21h bx

;是$结束符吗? ;不是,继续输入 ;对非数字字符进行计数

input bx, buff bx

;是$结束符,则转去显示 ;小于0是非数字字符 ;大于9是非数字字符 ;个数+1

;16进制数显示程序段(省略)

disp

next next count next

lea

bx, buff

;从键盘输入一个字符的功能调用

segment db ends

50 dup (‘ ’) dw 0

mov ah, 09h

;--------------------------------------------------------------------------

mov count, 0

mov ah, 01

mov [bx], al inc jnz lea next: inc jz jb ja

cmp al, ‘$’

mov cl, [bx]

cmp cl, ‘$’ cmp cl, 30h cmp cl, 39h inc jmp disp:

5.12 有一个首地址为mem的100d字数组,试编制程序删除数组中所有为0的项,并将后续项向前压缩,

最后将数组的剩余部分补上0。 答:程序如下: dseg mem dseg cseg main start: sub push ax mov ax, dseg mov ds, ax begin:

mov si, (100-1)*2

;(si)指向mem的末元素的首地址

mov bx, -2 mov cx, 100 comp: jz

cons finish

;比较完了,已无0则结束

;到了最后单元码?

add bx, 2

cmp mem [bx], 0 loop comp jmp cons: cons1: jae

mov di, bx

cmp di, si nomov

;地址指针的初值

;给ds赋值

segment

dw 100 dup (?) ends segment proc far push ds ax, ax

;设置返回dos

;--------------------------------------------------------------------------

assume cs: cseg, ds: dseg

mov ax, mem [di+2] ;后面的元素向前移位 mov mem [di], ax add di, 2 jmp nomov: finish: main cseg end start

5.13 在string到string+99单元中存放着一个字符串,试编制一个程序测试该字符串中是否存在数字,如有则把cl的第5位置1,否则将该位置0。 答:程序如下: dseg string db dseg

segment 100 dup (?) ends cons1

mov word ptr [si], 0 ;最后单元补0 endp ends

;以上定义代码段

ret

loop comp

;--------------------------------------------------------------------------

;--------------------------------------------------------------------------

cseg main start: sub push ax

segment proc far push ds ax, ax

;设置返回dos

assume cs: cseg, ds: dseg

mov ax, dseg mov ds, ax begin: repeat: jb ja or

mov si, 0

;(si)作为地址指针的变化值

mov cx, 100

mov al, string [si] go_on go_on cl, 20h exit inc

si

;不存在数字把cl的第5位置0

;存在数字把cl的第5位置1

cmp al, 30h cmp al, 39h

;给ds赋值

jmp go_on:

loop repeat and cl, 0dfh exit: main cseg end start

5.14 在首地址为table的数组中按递增次序存放着100h个16位补码数,试编写一个程序把出现次数最多的数及其出现次数分别存放于ax和cx中。 答:程序如下: dseg table data count dseg cseg main start: sub push ax mov ax, dseg mov ds, ax

;给ds赋值

segment

dw 100h dup (?) dw ? dw 0 ends segment proc far push ds ax, ax

;设置返回dos

;数组中的数据是按增序排列的

ret endp ends

;以上定义代码段

;--------------------------------------------------------------------------

;--------------------------------------------------------------------------

assume cs: cseg, ds: dseg

begin: mov si, 0 next: comp: jne inc

mov cx, 100h ;循环计数器

mov dx, 0 addr dx

cmp table [si], ax

;计算一个数的出现次数

mov ax, table [si]

add si, 2

cmp dx, count done

;目前此数出现的次数最多,记下次数 ;记下此数

;准备取下一个数

;出现最多的次数存入(cx) ;出现最多的数存入(ax)

;此数出现的次数最多吗?

loop comp addr: jle

mov count, dx mov data, ax done:

loop next

mov cx, count mov ax, data ret main cseg end start

endp ends

;以上定义代码段

;--------------------------------------------------------------------------

5.15 数据段中已定义了一个有n个字数据的数组m,试编写一程序求出m中绝对值最大的数,把它放在数据段的m+2n单元中,并将该数的偏移地址存放在m+2(n+1)单元中。 答:程序如下: dseg n m

segment

equ 100h dw n dup (?)

;m+2n单元 ;m+2(n+1)单元

;假设n=100h

data addr dseg cseg main start: sub push ax

dw ? dw ? ends segment proc far push ds ax, ax

;--------------------------------------------------------------------------

assume cs: cseg, ds: dseg

;设置返回dos

mov ax, dseg mov ds, ax begin: lea

mov cx, n

;循环计数器

di, m

;取第一个数

;记下绝对值最大的数的地址

;给ds赋值

mov ax, [di] mov addr, di

本文来源:https://www.bwwdw.com/article/6ont.html

Top