汇编语言跳转指令 DEC指令 要把数转为二进制吗

第 1 章 汇编语言基础知识〔习题 1.1〕 简述计算机系统的硬件组成及各部分作用。 〔解答〕 CPU:包括运算器、控制器和寄存器组。运算器执行所有的算术和逻辑运算;控制器 负责把指指令逐条从存储器中取出,经译码分析后向机器发出各种控制命令,并正确完成 程序所要求的功能;寄存器组为处理单元提供所需要的数据。 存储器:是计算机的记忆部件,它用来存放程序以及程序中所涉及的数据。 外部设备:实现人机交换和机间的通信。 〔习题 1.2〕 明确下列概念或符号: 主存和辅存,RAM和ROM,存储器地址和I/O端口,KB、MB、GB和TB 〔解答〕 主存又称内存是主存储器的简称,主存储器存放当前正在执行的程序和使用的数据, CPU可以直接存取,它由半导体存储器芯片构成其成本高、容量小、但速度快。辅存是辅 助存储器的简称,辅存可用来长期保存大量程序和数据,CPU需要通过I/O接口访问,它由 磁盘或光盘构成,其成本低、容量大,但速度慢。 RAM是随机存取存储器的英语简写,由于CPU可以从RAM读信息,也可以向RAM写入 信息,所以RAM也被称为读写存储器,RAM型半导体存储器可以按地址随机读写,但这类 存储器在断电后不能保存信息;而ROM中的信息只能被读出,不能被修改,ROM型半导体 通常只能被读出,但这类存储器断电后能保存信息。 存储器由大量存储单元组成。为了区别每个单元,我们将它们编号,于是,每个存储 单元就有了一个存储地址,I/O接口是由一组寄存器组成,为了区别它们,各个寄存器进行 了编号,形成I/O地址,通常称做I/O端口。 KB是千字节、MB是兆字节、GB是吉字节和TB是太字节,它们都是表示存储器存储单 元的单位。 〔习题 1.3〕 什么是汇编语言源程序、汇编程序、目标程序? 〔解答〕 用汇编语言书写的程序就称为汇编语言源程序; 完成汇编工作的程序就是汇编程序; 由汇编程序编译通过的程序就是目标程序。 〔习题 1.4〕 汇编语言与高级语言相比有什么优缺点? 〔解答〕 汇编语言与高级语言相比的优点:由于汇编语言本质就是机器语言,它可以直接地、 有效地控制计算机硬件,因而容易产生运行速度快,指令序列短小的高效目标程序,可以 直接控制计算机硬件部件,可以编写在―时间‖和―空间‖两方面最有效的程序。 汇编语言与高级语言相比的缺点:由于与处理器密切相关导致通用性差、可移植性差, 汇编语言功能有限,又涉及寄存器、主存单元等硬件细节,编写汇编语言比较繁琐,调试 起来也比较困难,编译程序产生的目标程序往往比较庞大、程序难以优化,运行速度慢。1 〔习题 1.5〕 将下列十六进制数转换为二进制和十进制表示 (1)FFH (2)0H (3)5EH (5)2EH (6)10H (7)1FH 〔解答〕 (1)FFH B (2)0H 0B (3)5EH 1011110B (4)EFH B (5)2EH 101110B (6)10H 10000B (7)1FH 11111B (8)ABH B 〔习题 1.6〕 将下列十进制数转换为BCD码表示 (1)12 (2)24 (3)68 (5)128 (6)255 (7)1234 〔解答〕 (1)12
(7) (8) 〔习题 1.7〕 将下列BCD码转换为十进制数 (1) (2) (5) (6) 〔解答〕 (1)91 (2)89 (3)36 (4)90 (5)08 (6)97 (7)81 (8)02(4)EFH (8)ABH 255D 0D 94D 239D 46D 16D 31D 171D(4)127 (8)2458(3) (7)(4) (8)〔习题 1.8〕 将下列十进制数分别用8位二进制数的原码、反码和补码表示2 (1)0 (5)126 〔解答〕 (1)0 +0 -0 (2)-127 (3)127 (4)-57 (5)126 (6)-126 (7)-128 (8)68(2)-127 (6)-126 11 10 00(3)127 (7)-128 (4)-57 (8)6811 0111 10 00〔习题 1.9〕 完成下列二进制数的运算 (1) (2) (3) (4)01 (5)1011 ? 1001 (6)1011 ? 1001 (7)?1011 (8)1011 ? 1001 〔解答〕 (1)=10100 (2)=0010 (3)=1100011 (4)01=10100,余数 1000 (5)1011 ?
(6)1011 ?
(7)? (8)1011 ?
〔习题 1.10〕 数码0~9、大写字母A~Z、小写字母a~z对应的ASCII码分别是多少?ASCII码为0dh、 0ah对应的是什么字符? 〔解答〕 数码0~9:30H~39H 大写字母A~Z:41H~5AH 小写字母a~z:61H~7AH ASCII码为0dh、0ah分别对应回车和换行控制字符。 〔习题 1.11〕 计算机中有一个―‖编码, 如果把它认为是无符号数, 它是十进制什么数?如果 认为它是BCD码,则表示什么数?又如果它是某个ASCII码,则代表哪个字符? 〔解答〕 十进制无符号数:B=61H=97 BCD码:61 ASCII码:a3 〔习题 1.12〕 简述Intel 80x86系列微处理器在指令集方面的发展。 〔解答〕 1978年Intel, 正式推出了16位8086CPU, 1979年Intel推出了准16位微处理器8088, 随后, Intel推出了,指令系统比8086指令系统新增了若干条实用的指令, 涉及堆栈操作、移位指令、过程指令和边界检测及乘法指令,1982年Intel推出80286 CPU, 80286指令系统包括全部80186指令及新增的保护指令15条,其中有些保护方式在实方式下 也可以使用,1985年,Intel80x86推出微处理器地进入第三代80386 CPU, 80386指令系统 在兼容原来16位指令系统的基础上,全面升级为32位,还新增了有关位操作、条件设置指 令以及控制、调试和测试寄存器的传送指令等,1989年,Intel推出了80486CPU,80486将浮 点处理单元FPU集成进来,还采用了精简指令集计算机技术RISC和指令流水线方式,还新 增了用于多处理器和内部Cache操作的6条指令,1993年Intel制成了俗称586的微处理器,取 名Pentium。Pentium仍为32位结构,地址总线为32位,对常用的简单指令用硬件实现,重新 设计指令的微代码等, Pentium新增了一条8字节比较交换指令和一条处理器识别指令, 以及 4条系统专用指令,1996年推出了MMX Pentium,新增了57条多媒休指令,1995年Intel推出 Pentium Pro新增了3条指令,1999年推出了PentiumⅢ新增了70条SSE指令,2000年推出的 Pentium4新增了76条SSE2指令 〔习题 1.13〕 什么是DOS和ROM-BIOS? 〔解答〕 DOS是Diskette Operating system的缩写, 意思是磁盘操作系统,DOS主要是面向磁盘的系 统软件,说得简单些,就是人与机器的一座桥梁,是罩在机器硬件外面的一层―外壳‖,是 年的个人电脑上使用的一种主要的操作系统。 BIOS (Basic Input/Output System) 即基本输入输出系统,通常是固化在只读存储器(ROM)中,所以又称为ROM-BIOS。它 直接对计算机系统中的输入、输出设备进行设备级、硬件级的控制,是连接软件程序和硬 件设备之间的枢纽。ROM-BIOS是计算机系统中用来提供最低级、最直接的硬件控制的程 序。 〔习题 1.14〕 简述PC机最低1MB主存空间的使用情况。 〔解答〕 (1)基本RAM区(00000H—9FFFFH)该区共640KB,由DOS进行管理。在这个区域中 操作系统要占用掉一部分低地址空间,其它则向用户程序开放。 (2) 保留区RAM (A0000H--BFFFFFH) 该区为系统安排的―显示缓冲存储区‖, 共126KB, 是显卡上的芯片提供支持,用于存放屏幕显示信息。但这部分地址空间实际上并没有全部 使用。 (3)扩展区ROM(C0000H--DFFFFH)该区128KB,由接口卡上的芯片提供支持,用 于为系统不直接支持的外设安排设备驱动程序。用户固化的程序就可[安排在这一段,系统 的会对它进行确认和连接。 (4)系统区ROM(E0000H--FFFFFH)该区共128KB,由系统占用,它主要提供 ROM--BIOS程序,基本输入输出程序BIOS,是操作系统的重要组成部分,主要用来驱动输 入输出设备, 也负责系统的上电检测, 磁盘引导等初始化操作, 在ROM--BIOS中还有CMOS 微机设置程序以及使用的字符图符信息等内容。4 〔习题 1.15〕 罗列8086CPU的8个8位和16位通用寄存器,并说明各自的作用。 〔解答〕 (1)数据寄存器:AX称为累加器,使用频度最高,用于算术、逻辑运算以及与外设 传送信息等;BX称为基址寄存器,常用做存放存储器地址;CX称为计数器,作为循环和串 操作等指令中的隐含计数器;DX称为数据寄存器,常用来存放双字长数据的高16位,或存 放外设端口地址。 (2)指针及变址寄存器包括SI,DI,BP,S P,四个寄存器,常用于存储器寻址时提供地址。 SI是源变址寄存器,DI是目的变址寄存器,一般与DS联用确定数据段和附加段中某一存储 单元地址,在串指令中,SI与DS联用、DI和ES联用,分别寻址数据段和附加段;同时,在 串指令中,SI和DI还都具有自动增量或减量的功能。S P,为堆栈指针寄存器,指示栈顶的偏 移地址;BP为基地址指针寄存器,表示堆栈段中的基地址。S P与BP寄存器均可与SS段寄存 器联合使用以确定堆栈段中的存储单元地址。 〔习题 1.16〕 什么是标志,它有什么用途?状态标志和控制标志有什么区别?画出标志寄存器 FLAGS,说明各个标志的位置和含义。 〔解答〕 标志用于反映指令执行结果或控制指令执行形式。它是汇编语言程序设计中必须特别 注意的一个方面,状态用来记录运行的结果的状态信息,许多指令的执行都将相应地设置 它,控制标志位可由程序根据需要用指令设置,用来控制处理器执行指令的方式 15 14 13 12 11 10 9 8 7 6 5 4 3 2 1 0 OF DF IF TF SF ZF AF SF CF CF是进位标志;ZF 是零标志;SF是符号标志;PF奇偶标志;OF溢出标志;AF辅助进 位标志;DF方向标志;IF中断允许标志;TF陷阱标志。 〔习题 1.17〕 举例说明CF和OF标志的差异。 〔解答〕 溢出标志OF和进位标志CF是两个意义不同的标志。 进位标志表示无符号数运算结果是否超出范围,运算结果仍然正确;溢出标志表示有 符号数运算结果是否超出范围,运算结果已经不正确。 例1:3AH + 7CH=B6H 无符号数运算:58+124=182,范围内,无进位 有符号数运算: 58+124=182 ,范围外,有溢出 例2:AAH + 7CH=(1)26H 无符号数运算:170+124=294,范围外,有进位 有符号数运算:-86+124=28 ,范围内,无溢出 〔习题 1.18〕 字和双字在存储器中如何存放, 什么是―小端方式‖?对字和双字存储单元, 什么是它们 的对齐地址?为什么要对齐地址? 〔解答〕 字或双字在存储器中占相邻的2个或4个存储单元;存放时,低字节存入低地址,高字 节存入高地址;字或双字单元的地址用它的低地址来表示。80x86处理器采用的这种―低对 低,高对高‖的存储形式,被称为―小端方式‖;将字单元安排在偶地址,双字节单元安排在5 模4地址,被称为―地址对齐方式‖因为对于不对齐地址的数据,处理器访问时,需要额外的 访问时间,所以通常应该将数据的地址对齐,以取得较高的存取速度。 〔习题 1.19〕 什么是8086中的逻辑地址和物理地址?逻辑地址如何转换成物理地址?请将如下逻辑 地址用物理地址表达: (1)FFFFh:0 (2)40h:17h (3)h (4)B821h:4567h 〔解答〕 在8086处理器中,对应每个物理存储单元都有一个唯一的20位编号,就是物理地址, 从00000H~FFFFFH。 在8086内部和用户编程时,采用的段基地址 : 段内偏移地址形式称为逻辑地址。 将逻辑地址中的段地址左移二进制4位(对应16进制是一位,即乘以16),加上偏移地 址就得到20位物理地址 如下逻辑地址用物理地址表达: (1)FFFFh:0=FFFF0H (2)40h:17h=00417H (3)h=24500H (4)B821h:4567h=BC777H (不要算错) 〔习题 1.20〕 8086有哪4种逻辑段,各种逻辑段分别是什么用途? 〔解答〕 代码段(Code Segment)用来存放程序的指令序列。处理器利用CS : IP取得下一条要执 行的指令。 堆栈段(Stack Segment)确定堆栈所在的主存区域。处理器利用SS : SP操作堆栈中的 数据。 数据段(Data Segment)存放当前运行程序所用的数据。处理器利用DS : EA存取数据 段中的数据。 附加段(Extra Segment)是附加的数据段,也用于数据的保存。处理器利用ES : EA存 取数据段中的数据。 〔习题 1.21〕 数据的默认段是哪个,是否允许其他段存放数据?如果允许, 如何实现,有什么要求? 〔解答〕 数据的默认段是安排在数据段,也经常安排在附加段,尤其是串操作的目的区必须是 附加段,允许其它段存放数据,数据的存放比较灵活的,实际上可以存放在任何一种逻辑 段中,这时,只要明确指明是哪个逻辑段就可以了。 〔习题 1.22〕 什么是操作码、操作数和寻址方式?有哪三种给出操作数的方法? 〔解答〕 操作码说明计算机要执行哪种操作,它是指令中不可缺少的组成部分,操作数是指令 执行的参与者,也是各种操作的对象,我们把寻找数的方式叫做操作数的寻址方式。给出 操作数的三种方法是直接给出,间接给出,隐藏操作数方式给出。 〔习题 1.23〕 什么是有效地址EA?8086的操作数如果在主存中,有哪些寻址方式可以存取它?6 〔解答〕 DS存放数据段的段地址,存储器中操作数的偏移地址则由各种主存方式得到,称之为 有效地址EA。8086的操作数如果在主存中,可以存取它的寻址方式有直接寻址方式、寄存 器间接寻址方式、寄存器相对寻址方式、基址变址寻址方式、相对基址变址寻址方式。 〔习题 1.24〕 说明下列指令中源操作数的寻址方式?如果BX=2000H,DI=40H,给出DX的值或有 效地址EA的值。 (1) mov dx,[1234h] (2) mov dx,1234h (3) mov dx,bx (4) mov dx,[bx] (5) mov dx,[bx+1234h] (6) mov dx,[bx+di] (7) mov dx,[bx+di+1234h] 〔解答〕 (1)直接寻址,EA=1234H (2)立即数寻址,DX=1234H (3)寄存器寻址,DX=2000H (4)间接寻址,EA=2000H (5)相对寻址,EA=3234H (6)基址变址寻址,EA=2040H (7)相对基址变址寻址,EA=3274H第 2 章 8086 的指令系统〔习题 2.1〕 已知DS=2000H、BX=0100H、SI=0002H,存储单元[20100H]~[20103H]依次存放12 34 56 78H,[21200H]~[21203H]依次存放2A 4C B7 65H,说明下列每条指令执行完后AX寄 存器的内容。 (1) mov ax,1200h (2) mov ax,bx (3) mov ax,[1200h] (4) mov ax,[bx] (5) mov ax,[bx+1100h] (6) mov ax,[bx+si] (7) mov ax,[bx][si+1100h] 〔解答〕 (1)AX=1200H (2)AX=0100H (3)AX=4C2AH ;偏移地址=bx=0100h (4)AX=3412H ;偏移地址=bx=0100h (5)AX=4C2AH ;偏移地址=bx+h (6)AX=7856H ;偏移地址=bx+si=h=0102h (7)AX=65B7H ;偏移地址=bx+si+h+h=1202h7 〔习题 2.2〕 指出下列指令的错误 (1) mov cx,dl (2) mov ip,ax (3) mov es,1234h (4) mov es,ds (5) mov al,300 (6) mov [sp],ax (7) mov ax,bx+di (8) mov 20h,ah 〔解答〕 (1)两操作数类型不匹配 (2)IP指令指针禁止用户访问 (3)立即数不允许传给段寄存器 (4)段寄存器之间不允许传送 (5)两操作数类型不匹配 (6)目的操作数应为[ SI ] (7)源操作数应为 [BX+DI] (8)立即数不能作目的操作数 〔习题 2.3〕 已知数字0 ~ 9对应的格雷码依次为:18H、34H、05H、06H、09H、0AH、0CH、11H、 12H、14H,它存在于以table为首地址(设为200H)的连续区域中。请为如下程序段的每条 指令加上注释,说明每条指令的功能和执行结果。lea bx,table mov al,8 xlat〔解答〕lea bx,table mov al,8 xlat ;获取table的首地址,BX=200H ;传送欲转换的数字,AL=8 ;转换为格雷码,AL=12H P35〔习题 2.4〕 什么是堆栈,它的工作原则是什么,它的基本操作有哪两个,对应哪两种指令? 〔解答〕 堆栈是一种按―先进后出‖原则存取数据的存储区域,位于堆栈段中,使用SS段寄存器 记录其段地址;它的工作原则是先进后出;堆栈的两种基本操作是压栈和出栈,对应的指 令是PUSH和POP。 〔习题 2.5〕 已知SS = 2200H、SP = 00B0H,画图说明执行下面指令序列时,堆栈区和SP的内容如 何变化?mov ax,8057h push ax 8 mov ax,0f79h push ax pop bx pop [bx]〔解答〕mov ax,8057h push ax mov ax,0f79h push ax pop bx pop [bx] ;bx=0f79DS:[0f79h]=8057h220B0H 80h 57h 0fh 79hSP=00B0H22000H SS=2200H〔习题 2.6〕 给出下列各条指令执行后AL值,以及CF、ZF、SF、OF和PF的状态:mov al,89h add al,al add al,9dh cmp al,0bch sub al,al dec al inc al〔解答〕mov al,89h add al,
+001 0010 ; AL=89 AL=12h CF 1 ZF 0 SF 0 OF 1 PF 19 add al,9
+ 10 1111 cmp al,0
1111; AL=0afh 00101; AL=0afh 10101; AL=00h01 00 10 01 1; AL=0ffh 0; AL=00h01001〔习题 2.7〕 设X、Y、Z均为双字数据,分别存放在地址为X、X+2;Y、Y+2;Z、Z+2的存储单元 中,它们的运算结果存入W单元。阅读如下程序段,给出运算公式。mov ax,X mov dx,X+2 add ax,Y adc dx,Y+2 add ax,24 adc dx,0 sub ax,Z sbb dx,Z+2 mov W,axmov W+2,dx 〔解答〕10 W=X+Y+24-Z 〔习题 2.8〕 请分别用一条汇编语言指令完成如下功能: (1)把BX寄存器和DX寄存器的内容相加,结果存入DX寄存器。 (2)用寄存器BX和SI的基址变址寻址方式把存储器的一个字节与AL寄存器的内容相 加,并把结果送到AL中。 (3) 用BX和位移量0B2H的寄存器相对寻址方式把存储器中的一个字和CX寄存器的内 容相加,并把结果送回存储器中。 (4)用位移量为0520H的直接寻址方式把存储器中的一个字与数3412H相加,并把结 果送回该存储单元中。 (5)把数0A0H与AL寄存器的内容相加,并把结果送回AL中。 〔解答〕 (1)ADD DX,BX (2)ADD AL,[BX+SI] (3)ADD [BX+0B2H],CX (4)ADD WORD PTR [0520H],3412H (5)ADD AL,0A0H 〔习题 2.9〕 设X、Y、Z、V均为16位带符号数,分别装在X、Y、Z、V存储单元中,阅读如下程序 段,得出它的运算公式,并说明运算结果存于何处。mov ax,X imul Y mov cx,ax mox bx,dx mov ax,Z cwd add cx,cx=Z的低16位+X*Y的低16位 adc bx,bx=Z的高16位+X*Y的高16位+低位进位 sub cx,540 ;cx=Z的低16位+X*Y的低16位-540 sbb bx,0 mov ax,V cwd sub ax,ax=V的低16位-(Z的低16位+X*Y的低16位-540) sbb dx,dx=V的高16位-(Z的高16位+X*Y的高16位+低位进位-低位借位)-低位借位 idiv X ;/X ;bx=Z的高16位+X*Y的高16位+低位进位-低位借位 ;ax=V ;ax=X ;DX.AX=X*Y ;cx=X*Y的低16位 ;bx=X*Y的高16位 ;ax=Z〔解答〕11 [V-(X*Y+Z-540)]/X AX存商,DX存余数 〔习题 2.10〕 指出下列指令的错误: (1) xchg [si],30h (2) pop cs (3) sub [si],[di] (4) push ah (5) adc ax,ds (6) add [si],80h (7) in al,3fch (8) out dx,ah 〔解答〕 (1)xchg的操作数不能是立即数 (2)不应对CS直接赋值 (3)两个操作数不能都是存储单元 (4)堆栈的操作数不能是字节量 (5)adc的操作数不能是段寄存器 (6)没有确定是字节还是字操作 (7)in不支持超过FFH的直接寻址 (8)out只能以AL/AX为源操作数 〔习题 2.11〕 给出下列各条指令执行后的结果,以及状态标志CF、OF、SF、ZF、PF的状态。mov ax,1470h and ax,ax or ax,ax xor ax,ax not ax test ax,0f0f0h〔解答〕mov ax,1470h and ax,11 0000 or ax,ax xor ax,ax not ax test ax,0f0f0h ;AX=1470H 0 ;AX=0000H 0 ;AX=FFFFH 0 ;AX=FFFFH 0 0 1 1 0 0 0 0 1 0 0 0 0 0 1 1 1 ;AX=1470H CF ;AX=1470H 0 ZF 0 SF 0 OF 0 PF 0注意:MOV和NOT指令不影响标志位;其他逻辑指令使CF=OF=0,根据结果影响其 他标志位。12 〔习题 2.12〕 假设例题2.32的程序段中,AX = 08H,BX = 10H,请说明每条指令执行后的结果和各 个标志位的状态。 〔解答〕指令 mov si,ax shl si,1 add si,ax mov dx,bx mov cl,03h shl dx,cl sub dx,bx add dx, 执行结果 ; SI=AX=0008H ; SI=2*AX=0010H ; SI=3*AX=0018H ; DX=BX=0010H ; CL=03H ; DX=8*BX=0080H ; DX=7*BX=0070H ; DX=7*BX+3*AX=0088H CF OF SF ZF PF 0 0 0 0 0 0 0 0 0 0 0 u 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 1 1 0 0 1注意:逻辑左移N次相当于无符号整数乘以2的N次方,逻辑右移N次相当于无符号整数 除以2的N次方。移位指令根据移位的数据设置CF,根据移位后的结果影响SF,ZF,PF。在 进行一位移位时, 根据最高符号位是否改变设置OF, 如改变则OF=1。 另外, 程序注释用―u‖ 表示标志无定义(不确定),―-‖表示无影响。 〔习题 2.13〕 编写程序段完成如下要求: (1)用位操作指令实现AL(无符号数)乘以10 (2)用逻辑运算指令实现数字0 ~ 9的ASCII码与非压缩BCD码的互相转换 (3)把DX.AX中的双字右移4位 〔解答〕 (1);不考虑进位 mov bl,al mov cl,3 shl al,cl add al,bl add al,bl ;考虑进位xor ah,ah mov bx,ax mov cl,3 shl ax,cl add ax,bx add ax,shl bx,1 ;*8 ;shl bl,1(2) 数字0~9的ASCII码是:30h~39h 非压缩BCD码的0~9是:00h~09h 方法一:and al,0实现ASCII到非压缩BCD码的转换13 or al,30h;实现非压缩BCD码到ASCII的转换 ;求反D5D4位,其他不变方法二:xor al,30h ;即高4位为3,则变为0;高4位为0,则变为3(3)mov cl,4 again: shr dx,1 rcr ax,1实现逻辑右移 ;采用“sar dx,1”,则实现算术右移〔习题 2.14〕 已知AL = F7H (表示有符号数-9) 分别编写用SAR和IDIV指令实现的除以2的程序段, , 并说明各自执行后,所得的商是什么? 〔解答〕 (1)用sar编写mov al,0f7h sar al,1 ; -9送AL
; 结果:AL=B=0FBH 即-5 ; -9送 字节符号扩展位字 ; 注意除数不可为立即数 ; 结果:商为al=fch (-4) ; 余数:ah=ffh (-1)(2)用idiv编写mov al,0f7h cbw mov bl,2 idiv bl结论:符号数的除法用idiv 准确。14 〔习题 2.15〕 已知数据段500h ~600h处存放了一个字符串,说明下列程序段执行后的结果:mov si,600h mov di,601h mov ax,ds mov es,ax mov cx,256 std rep movsb〔解答〕 DI=601H SI=600H500H 单元该字符串从DS:500H开始的256个字节上移一个字节DS=ES 〔习题 2.16〕 说明下列程序段的功能cld mov ax,0fefh mov cx,5 mov bx,3000h mov es,bx mov di,2000h rep stosw〔解答〕 AH 0fefh 0fefh 0fefh 0fefh 0fefh H15 将es:di (即h或32000h)开始的5个单元的内容置为0fefh。 〔习题 2.17〕 指令指针IP是通用寄存器还是专用寄存器?有指令能够直接它赋值吗?哪类指令的执 行会改变它的值? 〔解答〕 指令指针IP不是通用寄存器,不可直接赋值,属于专用寄存器。有且仅有循环、转移、 子程序调用和返回、中断类等指令可以改变它的值。 〔习题 2.18〕 控制转移类指令中有哪三种寻址方式? 〔解答〕 控制转移类指令的寻址方式:相对寻址、直接寻址方式和间接寻址方式(又可以分成 寄存器和存储器间接寻址)。 〔习题 2.19〕 什么是短转移short jump、 近转移near jump和远转移far jump?什么是段内转移和段间转 移?8086有哪些指令可以实现段间转移? 〔解答〕 短转移:指段内-128~127之间的转移,位移量用一个字节表示 近转移:指段内±32K之间的转移,位移量用一个字表示 远转移:指段间1MB范围的转移 段内转移:指在同一个代码段内的转移,可以是短转移或者近转移 段间转移:指转移到另外一个代码段,就是远转移 CPU的JMP、CALL和INT n指令可以实现段间转移 〔习题 2.20〕 8086的条件转移指令的转移范围有多大?实际编程时,你如何处理超出范围的条件转 移? 〔解答〕 8086的条件转移的转移范围:在当前指令地址的 +127~-128之内。 如条件转移的转移范围超出此范围,可在此范围内安排一条无条件转移,再转移到范 围外的目标地址。 〔习题 2.21〕 假设DS=2000H,BX=1256H,SI=528FH,位移量TABLE=20A1H,[232F7H]=3280H, [264E5H]=2450H,试问执行下列段内间接寻址的转移指令后,转移的有效地址是什么? (1) JMP BX (2) JMP TABLE[BX] (3) JMP [BX][SI] 〔解答〕 (1)转移的有效地址EA= BX=1256H (2)转移的有效地址EA= [DS:20A1H+1256H]=[232F7]=3280H (3)转移的有效地址EA= [DS:FH]=264E5H=2450H16 〔习题 2.22〕 判断下列程序段跳转的条件(1) xor ax,1e1eh je equal (2)test al,b jnz there (3) cmp cx,64h jb there〔解答〕 (1)AX=1e1eh(异或后为0) (2)AL的D0或D7至少有一位为1 (3)CX(无符号数)< 64h 〔习题 2.23〕 设置CX = 0,则LOOP指令将循环多少次?例如:mov cx,0 delay: loop delay〔解答〕 216次。 〔习题 2.24〕 假设AX和SI存放的是有符号数, DX和DI存放的是无符号数, 请用比较指令和条件转移 指令实现以下判断: (1)若DX & DI,转到above执行; (2)若AX & SI,转到greater执行; (3)若CX = 0,转到zero执行; (4)若AX-SI产生溢出,转到overflow执行; (5)若SI≤AX,转到less_eq执行; (6)若DI≤DX,转到below_eq执行。 〔解答〕 (1)若DX & DI,转到above执行cmp dx,di ja above ;=jnbe above(2)若AX & SI,转到greater执行 cmp ax,sijg greater ;=jnle greater(3)若CX = 0,转到zero执行cmp cx,0 jz zero cmp ax,= jcxz zero(4)若AX-SI产生溢出,转到overflow执行;(5)若SI≤AX,转到less_eq执行;cmp si, cmp ax,si 17 jle less_eq cmp di,dx jbe below_eq; jge less_ cmp dx, jae below_eq(6)若DI≤DX,转到below_eq执行。〔习题 2.25〕 有一个首地址为array的20个字的数组,说明下列程序段的功能。mov cx,20 mov ax,0 mov si,ax sum_loop: add ax,array[si] add si,2 loop sum_loop mov total,ax〔解答〕 将首地址为array得20个字的数组求和,并将结果存入 total 单元中。 〔习题 2.26〕 按照下列要求,编写相应的程序段: (1)起始地址为string的主存单元中存放有一个字符串(长度大于6),把该字符串中 的第1个和第6个字符(字节量)传送给DX寄存器。 (2)从主存buffer开始的4个字节中保存了4个非压缩BCD码,现按低(高)地址对低 (高)位的原则,将它们合并到DX中。 (3)编写一个程序段,在DX高4位全为0时,使AX = 0;否则使AX = -1。 (4)有两个64位数值,按―小端方式‖存放在两个缓冲区buffer1和buffer2中,编写程序 段完成buffer1-buffer2功能。 (5)假设从B800h : 0开始存放有100个16位无符号数,编程求它们的和,并把32位的 和保存在DX.AX中。 (6)已知字符串string包含有32KB内容,将其中的‘$‘符号替换成空格。 (7)有一个100个字节元素的数组,其首地址为array,将每个元素减1(不考虑溢出) 存于原处。 (8)统计以 ‘$‘ 结尾的字符串srting的字符个数。 〔解答〕 (1)解答:mov si,0 mov dl,string[si] mov si,5 mov dh,string[si] ;第6个字符送dh寄存器:mov dl,stirng[5] ;si清零 ;第一字节 ;第二字节 ;第1个字符送dl寄存器:mov dl,stirng[0](2)解答:xor si,si mov al,buffer[si] inc si mov ah,buffer[si]18 mov cl,4 shl ah,cl or al,ah mov dl,al inc si mov al,buffer[si] inc si mov ah,buffer[si] mov cl,4 shl ah,cl or al,ah mov dh,al ;BCD码移到高半字节 ;组合成压缩BCD码 ;存入dh寄.. ; test dh,0f0 mov ax,0 ; mov ax,0ffffh ;第四字节 ;第三字节 ;BCD码移到高半字节 ;组合成压缩BCD码 ;存入dl寄..(3)解答:test dx,0f000h jz next mov ax,-1 jmp again next: again: mov ax,0 ... mov ax, word ptr buffer1 sub ax, word ptr buffer2 mov dx, word ptr buffer1+2 sbb dx, word ptr buffer2+2 ;后减高16位,需减低16位的借位 ;先减低16位(4)解答:(5)解答:mov ax,0b800h mov ds,ax xor si,si xor dx,dx mov cx,99 mov ax,[si] again: inc si inc si add ax,[si] jnc noc inc dx noc: dec cx jnz cx,again ;加下一个数 ;无进位转 ;有进位dx=dx+1 ;次数-1 ;非0继续加 ;段地址 ;地址偏移量si=0 ;和的高字dx=0 ;加的次数 ;第一个数 ;指向下一个字单元(6)解答1:不使用串操作指令(更好)mov si,offset string mov cx,^15=8000h 19 again:cmp byte ptr [si], ?$? jnz next mov byte ptr [si], ? ?;?$? =24 不要采用 jz 进行分支 ;? ?=20 jnz againnext:inc si loop again(6)解答2:使用串操作指令mov di,offset string mov al,?$? mov cx,8000h cld again: scasb jnz next mov byte ptr es : [di-1], ? ? next: loop again mov si,offset array mov cx,100 again: dec byte ptr [si] inc si loop again(7)解答1:(7)解答2:xor si,si mov cx,100 again: dec array[si] si&--0 ; 循环次数(7)解答3:mov si,offset array mov di,si mov ax,ds mov es,ax mov cx,100 cld again: lodsb dec al stosb loop again(8)解答:xor si,si coun: cmp string[si],'$' 20 ;si&--0 je done inc si jmp coun done: ...〔习题 2.27〕 对下面要求,分别给出3种方法,每种方法只用一条指令。 (1)使CF=0 (2)使AX=0 (3)同时使AX=0和CF=0 〔解答〕 (1)解答: clc and ax,ax or ax,ax (2)解答: xor ax,ax and ax,0 mov ax,0 (3)解答: and ax,0 xor ax,ax sub ax,ax 〔习题 2.28〕 参照本习题的示意图,分析调用序列,画出每次调用及返回时的堆栈状态。其中CALL 前是该指令所在的逻辑地址;另外,段内直接调用指令的机器代码的字节数为3,段间直接 调用指令则为5个字节。 〔解答〕 主程序转子suba时段内调用:断点1为2000h:0400h+3, 转子是只将IP压栈。 suba转子subb时段间调用:断点2为2000h:0840h+5, 转子时须将cs段地址和IP压栈 suba转子subc时段内调用:断点3为2000h:0c021h+3,转子是只将IP压栈。 注:压栈时先修改sp再压入断点,弹栈时先弹出断点再修改sp。 〔习题 2.29〕 已知AX 、 BX存放的是4位压缩BCD表示的十进制数, 请说明如下子程序的功能和出口 参数。add al,bl daa xchg al,ah adc al,bh daa xchg al,ah ret21 〔解答〕 压缩BCD码加法:AX←AX+BX 出口参数:AX=BCD码和 〔习题 2.30〕 AAD指令是用于除法指令之前,进行非压缩BCD码调整的。实际上,处理器的调整过 程是:AL←AH×10+AL,AH←0。如果指令系统没有AAD指令,请用一个子程序完成这 个调整工作。 〔解答〕shl ah,1 mov dl,ah mov cl,2 shl ah,cl add ah,dl add al,ah xor ah,ah int 3 ;ah=2*a (设原ah=a) ;dl=2*设定移位次数 ;ah=8*ah=10*al=10*a+清零返回DOS注意: 入口:AX中存放有―和‖(两非压缩BCD码) 出口:AL中 已为调整后的二进制数 〔习题 2.31〕 解释如下有关中断的概念: (1) 内部中断和外部中断 (2) 单步中断和断点中断 (3) 除法错中断和溢出中断 (4) 中断向量号和中断向量表 〔解答〕 (1) 内部中断是由于8086CPU内部执行程序引起的程序中断; 外部中断是来自8086CPU 之外的原因引起的程序中断; (2)单步中断是若单步标志TF为1,则在每条指令执行结束后产生的中断;断点中断 是供调试程序使用的, 它的中断类型号为3通常调试程序时, 把程序按程序的任务分成几段, 然后,每段设一个段点; (3)除法错中断是在执行除法指令时,若除数为0或商超过了寄存器所能表达的范围 产生的中断;溢出中断是在执行溢出中断指令INTO时,若溢出标志OF为1时产生的中断; (4)中断向量号是 中断类型号;中断向量表是中断向量号与它所对应的中断服务程 序起始地址的转换表。 〔习题 2.32〕 试比较INT n和段间CALL指令、IRET和段间RET指令的功能。 〔解答〕 INT n 响应中断时,除象CALL保护断点外,还应保护FR;段间CALL指令用在主程序 中实现子程序的调用。IRET 返回时,除象RET恢复断点外,还应恢复FR;子程序执行完成 后,应返回主程序中继续执行,这一功能由RET指令完成。 〔习题 2.33〕 什么是系统功能调用?汇编语言中,它的一般格式是怎样的?22 〔解答〕 系统功能调用是用户在程序一级请示操作系统服务的一种手段,它不是一条简单的硬 指令,而是带有一定功能号的―访指令‖,它的功能并非由硬件直接提供,而是由操作系统操 作系统中的一段程序完成的,即由软件方法实现的 汇编语言中,它的一般格式是分如下四步进行: (1)在AH寄存器置系统功能调用号; (2)在指定的寄存器中设置 入口参数; (3)用INT21H(或ROM—BIOS的中断向量号)指令执行功能调用; (4)据出口参数分析功能调用运行情况。 〔习题 2.34〕 补充例2.40,当有溢出时显示―Error! Overflow!‖,无溢出时显示―OK‖。 〔解答〕okmsg errmsg db ?OK?, ?$? db ?Error ! Overflow !?, ?$? … mov ax,X sub ax,Y jo overflow mov dx,offset okmsg jmp next overflow: mov dx,errmsg next: mov ah,9 int 21h错误解答:mov ax,X sub ax,Y jo overflow mov dx,offset okmsg okmsg db ?OK?, ?$? mov dx,errmsg mov ah,9 int 21h overflow: errmsg db ?Error ! Overflow !?, ?$? mov dx,errmsg mov ah,9 int 21 错误2:缺少JMP指令 ;错误1:数据定义在代码中〔习题 2.35〕 补充例2.42,显示―1‖的个数;注意首先将个数转换为ASCII码。 〔解答〕and al,7fh mov dl,al ;使d7=0 ;转存于dl23 jnp next or al,80h next: again: xor bl,bl mov cx,8 shl dl,1 jnc desp inc bl desp: loop again or bl,30h mov ah,02h mov dl,bl int 21h ret;奇数个1转 ;偶数个1 使d7=1 ;纪录1的个数 dl=0 ;移位次数 ;逻辑左移1次 ;如cf=0 转 ;否则cf=1 个数加1 ;cx=cx-1 如cx不等于0,循环 ;个数变为ASCII码 ;在CRT上显示个数〔习题 2.36〕 先提示输入数字―Input Number:0 ~ 9‖,然后在下一行显示输入的数字,结束;如果不 是键入了0 ~ 9数字,就提示错误―Error!‖,继续等待输入数字。 〔解答〕; 数据段 str1 str2 db 'Input Number:0~9 : ',0dh,0ah,'$' db 'Error!',0dh,0ah,'$' ; 代码段 mov ah,09h mov dx,offset str1 int 21h getkey: mov ah,1 int 21h cmp al,'0' jb error cmp al,'9' ja error mov ah,02h mov dl,al int 21h ... error: mov ah,09h mov dx,offset str2 int 21 返回按键 ; 终止程序执行,返回DOS ; 出错,调用DOS 功能显示str2字符串 ; 大于9,出错处理 ; 调用DOS显示字符功能,显示该数字 ; 小于0,出错处理 ; 调用DOS功能 ; 显示str1字符串〔习题 2.37〕 从键盘输入一个字符串(不超过255个),将其中的小写字母转换成大写字母,然后按24 原来的顺序在屏幕上显示。 〔解答〕 思路:参考P67 例2.52用0ah号DOS系统功能调用,从键盘输入一个字符串,然后从键 盘缓冲区逐个取字符, 在―a‖与―z‖之间的字符为小写字母, 需要转换为大写字母 (减去20h) , 其他不变。;xt237.asm .model small .stack .data array array1 array2 db 255 db 0 db 255 dup('$') db 0dh,0ah,'$' .code .startup mov ah,0ah mov dx,offset array int 21h mov dx,offset array2 mov ah,09h int 21h mov bx,offset array1 again: mov al,[bx] cmp al,'$' jz done cmp al,'a' jb next cmp al,'z' ja next sub al,20h mov [bx],al next: done: inc bx jmp again mov dx,offset array1 mov ah,09h int 21h .exit 0 在a和z之间的字符才是小写字母,转换为大写 ; 保存到原位置 ; 小于a和大于z的字符不是小写字母 ; 回车换行 ; 键盘输入字符串〔习题 2.38〕 指令对状态标志的作用可以分成多种情况,例如无影响、无定义、按结果影响、特别25 说明的影响等,你能区别这些情况吗?分别用具体的指令来说明。 〔解答〕 指令对状态标志的影响 OF 指令 ADD/ADC/SUB/SBB/CMP/NEC/CMPS/SCAS x INC/DEC x MUL/IMUL # DIV/IDIV u AND/OR/XOR/TEST 0 SAL/SAR/SHL/SHR # ROL/ROR/RCL/RCR # SHLD/SHRD # XADD/CMPXCHG x 状态符号说明 符号 0 1 x # u 说明 标志位不受影响(没有改变) 标志位复位(置 0) 标志位置位(置 1) 标志位按定义功能改变 标志位指令的特定说明改变 标志位不确定(可能为 0,也可能为 1) SF x x u u x x x x ZF x x u u x x x x AF x x u u u u u x PF x x u u x x x x CF x # u 0 # # # x〔习题 2.39〕 8086指令系统分成哪6个功能组?各组主要包含什么指令,举例说明。 〔解答〕 8086指令系统分成的6个功能组是 (1)数据传送类令: ①通用数据传送指令如传送指令MOV、交换指令XCHG、换码指令XLAT ②堆栈操作指令如进栈指令PUSH出栈指令POP ③标志传送指令如标志寄存器传送;标志位操作 ④地址传送指令如有效地址传送指令LEA、指针传送指令LDS指针传送指令LES ⑤输入输出指令如输入指令IN、输出指令OUT (2)算术运算指令 ①加法指令如加法指令ADD、带进位加法指令ADC、增量指令INC ②减法指令如减法指令SUB、带借位指令SBB、减量指令DEC、求补指令NEG、比较指 令CMP ③乘法指令如无符号乘法指令MUL、有符号乘法指令IMUL ④除法指令如无符号数除法指令DIV、有符号数除法指令IDIV ⑤符号扩展指令如字节转换字指令CBW、字转换字节指令CWD ⑥十进制调整指令如压缩BCD码调整指令、非压缩BCD码调整指令 (3)位操作类指令 ①逻辑运算指令如逻辑指令AND、逻辑或指令OR、逻辑异或指令XOR、逻辑非指令 NOT、测试指令TEST ②移位指令如逻辑移位指令SHL和SHR、算术移位指令SAL和SAR26 ③循环移位指令如不带移位指令XOL和ROR、带移位指令RCL和RCR (4)控制转移类指令 ①无条件转移指令JMP ②条件转移指令如判断单个标志状态JZ/JE、 JNZ/JNE、 JS/JNS、 JP/JPE、 JNP/JPO、 JC/JB、 JNC/JNB/JAE,用于比较无符号数高低JB(JNAE)、JNB(JAE)、JBE(JNA)、JNBE(JA),用于 比较有符号数大小JL(JNGE)、JNL(JGE)、JLE(JNG)、JNLE(JG) ③循环指令LOOP ④子程序指令如子程序调用指令CALL、子程序返回指令RET ⑤中断指令如INT N (5)串操作类指令 ①串传送指令MOVS ②串存储指令STOS ③串读取指令LODS ④串比较指令CMPS ⑤串扫描指令SCAS ⑥重复前缀指令REP; (6)处理机控制类指令 ①空操作指令NOP ②段超越前缀指令SEG ③封锁前缀指令LOCK ④暂停指令HLT ⑤交权指令ESC ⑥等待指令WAIT。 〔习题 2.40〕 总结8086指令系统所采用的各种寻址方式,包括一般的数据寻址、外设数据寻址、堆 栈数据寻址、串操作数据寻址、转移指令目的地址的寻址等,并举例说明。 〔解答〕 1. 一般的数据寻址 立即数寻址方式如:mov al,05h 寄存器寻址方式如:mov bx,ax 存储器寻址方式 (1)直接寻址方式如:mov ax,[2000h] (2)寄存器间接寻址方式如:mov ax,es:[2000h] (3)寄存器相对寻址方式如:mov ax,[di+06h] (4)基址变址寻址方式如:mov ax,[bx+si] (5)相对基址变址寻址方式如:mov ax,[bx+si+06h] 2. 外设数据寻址 输入指令IN,如: in al,21h in ax,dx 输出指令OUT,如: out dx,al 3. 堆栈数据寻址 进栈指令PUSH,如:PUSH [2000H]27 出栈指令POP,如:POP [2000H] 4. 串操作数据寻址 5. 转移指令目的地址的寻址 直接寻址、间接寻址、相对寻址第 3 章 汇编语言程序格式〔习题 3.1〕 伪指令语句与硬指令语句的本质区别是什么?伪指令有什么主要作用? 〔解答〕 伪指令语句与硬指令语句的本质区别是能不能产生CPU动作; 伪指令的作用是完成对如存储模式、主存变量、子程序、宏及段定义等很多不产生CPU 动作的说明,并在程序执行前由汇编程序完成处理。 〔习题 3.2〕 什么是标识符,汇编程序中标识符怎样组成? 〔解答〕 为了某种需要,每种程序语言都规定了在程序里如何描述名字,程序语言的名字通常 被称为标识符; 汇编语言中的标识符一般最多由31个字母、数字及规定的特殊符号(如-,$,?,@) 组成,不能以数字开头。 〔习题 3.3〕 什么是保留字,汇编语言的保留字有哪些类型,并举例说明。 〔解答 保留字是在每种语言中规定了有特殊意义和功能的不允许再做其它用处的字符串;汇 编语言的保留字主要有硬指令助记、伪指令助记符、运算符、寄存器名以及预定义符号等。 汇编语言对大小写不敏感。如定义字节数和字符串的DB就是伪指令助记符。 〔习题 3.4〕 汇编语句有哪两种,每个语句由哪4个部分组成? 〔解答〕 汇编语句有执行性语句和说明性语句; 执行性语句由标号、硬指令助记符、操作数和注释四部分组成; 说明性语句由名字、伪指令助记符、参数和注释四部分组成 〔习题 3.5〕 汇编语言程序的开发有哪4个步骤,分别利用什么程序完成、产生什么输出文件。 〔解答〕 ⒈ 编辑 文本编辑程序 汇编语言源程序.asm ⒉ 汇编 汇编程序 目标模块文件.obj ⒊ 连接 连接程序 可执行文件.exe或.com ⒋ 调试 调试程序 应用程序 〔习题 3.6〕 将第2章习题2.36采用简化段定义格式编写成一个完整的源程序。 〔解答〕28 ;简化段定义格式 .model small .stack .data str1 str2 ; 定义程序的存储模式(小模式) ; 定义堆栈段(默认1024个字节) ; 定义数据段db 'Input Number:0~9 : ',0dh,0ah,'$' db 'Error!',0dh,0ah,'$' .cade .startup mov ah,09h mov dx,offset str1 int 21 定义代码段 ; 说明程序的起始点,建立ds,ss的内容。 ; 显示str1字符串getkey:mov ah,1 int 21h cmp al,'0' jb error cmp al,'9' ja error mov ah,02h mov dl,al int 21h .exit 0; 调用DOS功能; 小于0,出错处理 ; 大于9,出错处理 ; 调用DOS显示字符功能,显示该数字; 终止程序执行,返回DOS ; 出错,调用DOS 功能显示str2字符串error:mov ah,09h mov dx,offset str2 int 21h jmp getkey end; 返回按键 ; 汇编结束〔习题 3.7〕 将第2章习题2.37采用完整段定义格式编写成一个完整的源程序。 〔解答〕;xt307.asm stack stack data array array1 array2 data code segment dw 512 dup(?) ends segment db 255 db 0 db 255 dup('$') db 0dh,0ah,'$' ends segment 'code'29 assume cs:code, ds:data, ss:stack start: mov ax,data mov ds,ax mov ah,0ah mov dx,offset array int 21h mov dx,offset array2 mov ah,09h int 21h mov bx,offset array1 again: mov al,[bx] cmp al,'$' jz done cmp al,'a' jb next cmp al,'z' ja next sub al,20h mov [bx],al next: done: inc bx jmp again mov dx,offset array1 mov ah,09h int 21h mov ax,4c00h int 21h
在a和z之间的字符才是小写字母,转换为大写 ; 保存到原位置 ; 小于a和大于z的字符不是小写字母 ; 回车换行 ; 键盘输入字符串〔习题 3.8〕 区分下列概念: (1)变量和标号 (2)数值表达式和地址表达式 (3)符号常量和字符串常量 〔解答〕 (1) 变量是在程序运行过程中, 其值可以被改变的量;标号是由用户自定义的标识符, 指向存储单元,表示其存储内容的逻辑地址。 (2)数值表达式一般是由运算符连接的各种常数所构成的表达式,地址表达式是由名 字、标号以及利用各种的操作符形成的表达式。 (3)在程序中,为了使常量更便于使用和阅读,经常将一些常量用常量定义语句定义为 符号常量,被一对双引号括起来的若干个字符组成的字符序列被称为字符串常量。30 〔习题 3.9〕 假设myword是一个字变量,mybyte1和mybyte2是两个字节变量,指出下列语句中的错 误原因。 (1) mov byte ptr [bx],1000 (2) mov bx,offset myword[si] (3) cmp mybyte1,mybyte2 (4) mov al,mybyte1+mybyte2 (5) sub al,myword (6) jnz myword 〔解答〕 (1)1000超出了一个字节范围 (2)寄存器的值只有程序执行时才能确定,而offset是汇编过程计算的偏移地址,故无 法确定,改为lea bx,myword[si] (3)两个都是存储单元,指令不允许 (4)变量值只有执行时才确定,汇编过程不能计算 (5)字节量AL与字量myword,类型不匹配 (6)Jcc指令只有相对寻址方式,不支持间接寻址方式 〔习题 3.10〕 OPR1是一个常量,问下列语句中两个AND操作有什么区别? AND AL,OPR1 AND 0feh 〔解答〕 前者为―与‖操作硬指令助记符,可汇编成机器代码。 后者为逻辑运算符,在汇编时进行―与‖运算,产生具体数值。 〔习题 3.11〕 给出下列语句中,指令立即数(数值表达式)的值: (1) mov al,23h AND 45h OR 67h (2) mov ax,h (3) mov ax,NOT(65535 XOR 1234h) (4) mov al,LOW 1234h OR HIGH 5678h (5) mov ax,23h SHL 4 (6) mov ax,1234h SHR 6 (7) mov al,‘a‘ AND (NOT(‘a‘-‘A‘)) (8) mov al,‘H‘ OR b (9) mov ax,(76543 LT 32768) XOR 7654h 〔解答〕 注:对于逻辑运算,有关操作数可化为二进制数。 (1)67h (2)133h (3)1234h (4)76h (5)0234h (6)0048h (7)41h31 (8)68h (9)7654h 〔习题 3.12〕 为第2章例题2.54定义变量count、block、dplus和dminus。 〔解答〕 假设block开始的数据块有32个字节数据:16个正数+100 (64h)、16个负数 -48 (0d0h) 分别连续分布: block db 16 dup(100),16 dup(-48) ;也可以是任意字节数据,随意分布。 dplus db 32 dup(?) ;为正数预留存储空间 dminus db 32 dup(?) ;为负数预留存储空间 count equ 32 ;字节数 〔习题 3.13〕 为第2章例题2.55定义相应变量,并形成一个完整的汇编语言程序。 〔解答〕;lt239b.asm .model small .stack .data string1 string2 result db 'good morning !' ;两字符串可相同或不同,但字符数要求相同。 db 'Good morning !' db ? .code .startup mov ax,ds mov es,ax mov si,offset string1 mov di,offset string2 mov cx,count again: cmpsb jnz unmat dec cx jnz again mov al,0 jmp output unmat: mov al,0ffh .exit0 end output: mov result, al ;所有数据在同一个段,所以使es=ds ;预留结果字节 ;字符数count = 1432 〔习题 3.14〕 画图说明下列语句分配的存储空间及初始化的数据值: (1) byte_var DB ‘ABC‘,10,10h,‘EF‘,3 DUP(-1,?,3 DUP(4)) (2) word_var DW 10h,-5,‘EF‘,3 DUP(?) 〔解答〕 (1)从低地址开始,依次是(十六进制表达): 41 42 43 0a 10 45 46 ff - 04 04 04 ff - 04 04 04 (2)从低地址开始,依次是(十六进制表达): 10 00 FB FF 46 45 - - - - - - 〔习题 3.15〕 请设置一个数据段mydataseg,按照如下要求定义变量: (1) my1b为字符串变量:Personal Computer (2) my2b为用十进制数表示的字节变量:20 (3) my3b为用十六进制数表示的字节变量:20 (4) my4b为用二进制数表示的字节变量:20 (5) my5w为20个未赋值的字变量 (6) my6c为100的常量 (7) my7c表示字符串:Personal Computer 〔解答〕mydataseg segment my1b my2b my3b my4b my5w my6c my7c db ?Personal Computer? db 20 db 14h db b dw 20 dup(?) equ 100 ;my6c = 100 equ &Personal Computer& ;20h0404ff- 04mydataseg ends〔习题 3.16〕 分析例题3.2的数据段,并上机观察数据的存储形式。 〔解答〕 以字节为单位从低地址向高地址依次是: 16 00 12 FFH FFH FFH FFH 00 00 00 00 00 00 00 00 1 2 3 4 5 45H 23H 00 00 00 00 00 00 00 00 ?a‘ ?b‘ ?c‘ ?H‘ ?e‘ ?l‘ ?l‘ ?o‘ 13 10 ?$‘ 12个字符串‘month‘,每个字符串从低地址到高地址依次是:‘m‘ ?o‘ ?n‘ ?t‘ ?h‘ 25×4个字节未定义初值的存储单元,操作系统设置为033 〔习题 3.17〕 修改例题3.3, 现在用字定义伪指令dw、 字串传送指令movsw和字符串显示9号功能调用 实现。 〔解答〕.model small .stack .data source target dw h dw 40 dup(?),'$' .code .startup mov ax,ds mov es,ax cld mov si,offset source mov di,offset target mov cx,40 rep movsw mov si,0 mov dx,offset target mov ah,9 int 21h .exit 0 end〔习题 3.18〕 变量和标号有什么属性? 〔解答〕 段地址:表示变量和标号所在代码段的段地址; 偏移地址:表示变量和标号所在代码段的段内偏移地址; 类型:引用变量时,表示是字节、字、双字等数据量。引用该标号时,表示它所在同 一个段――near类型,还是另外一个段――far类型。 〔习题 3.19〕 设在某个程序中有如下片段,请写出每条传送指令执行后寄存器AX的内容: mydataVARW VARB VARDsegmentORG 100H DW H DB 3,4 ALIGN 4 DD H EVEN34 BUFF MESS BEGIN:DB 10 DUP(?) DB ?HELLO? MOV AX,OFFSET MESS MOV AX,TYPE BUFF+TYPE MESS+TYPE VARD MOV AX,SIZEOF VARW+SIZEOF BUFF+SIZEOF MESS MOV AX,LENGTHOF VARW+LENGTHOF VARD MOV AX,LENGTHOF BUFF+SIZEOF VARW MOV AX,TYPE BEGIN MOV AX, OFFSET BEGIN〔解答〕MOV AX, OFFSET MESS MOV AX, TYPE BUFF+TYPE MESS+TYPE VARD MOV AX,SIZEOF VARW+SIZEOF BUFF+SIZEOF MESS MOV AX,LENGTHOF VARW + LENGTHOF VARD MOV AX,LENGTHOF BUFF + SIZEOF VARW MOV AX,TYPE BIGIN MOV AX,OFFSET BEGIN ;AX=116H ;AX = 1+1+4 = 06H ;AX = 4+10+5 = 19 = 13H ;AX = 2+1 = 03H ;AX = 10+4 =14 = 0EH ;AX = FF02H (近) ;AX = 1BH〔习题 3.20〕 利用简化段定义格式,必须具有.MODEL语句。MASM定义了哪7种存储模式,TINY和 SMALL模式创建什么类型(EXE或COM)程序?设计32位程序应该采用什么模式? 〔解答〕 MASM定义的7种存储模式是TINY(微型模式、SMALL(小型模式)、COMPACT(紧 凑模式)、MEDIUM(中型模式)、LARGE(大型模式)、HUGE(巨大模式)、FLAT (平展模式);TINY用于创建COM类型程序、一般程序都可以选用SMALL模式;设计32 位的程序应该采用FLAT模式。 〔习题 3.21〕 源程序中如何指明执行的起始点?源程序应该采用哪个DOS功能调用,实现程序返回 DOS? 〔解答〕 源程序中运用STARTUP伪指令指明执行的起始点;源程序应该采用DOS功能调用的 4CH子功能实现程序返回DOS的。 〔习题 3.22〕 在SMALL存储模式下,简化段定义格式的代码段、数据段和堆栈段的缺省段名、定位、 组合以及类别属性分别是什么? 〔解答〕 段定义伪指令 段名 定位 组合 类别 组名 .CODE .DATA .DATA? .STACK _TEXT _DATA _BSS STACK WORD PUBLIC 'CODE' WORD PUBLIC 'DATA' DGROUP WORD PUBLIC 'BSS' DGROUP PARA STACK 'STACK' DGROUP35 〔习题 3.23〕 如何用指令代码代替.startup和.exit指令,使得例题3.1a能够在MASM 5.x下汇编通过? 〔解答〕 ;lt301a.asm(文件名) .model small .stack .data string db ?Hello,Everybody !‘,0dh,0ah.‘$‘ .code start: mov ax ,@data mov ds,ax mov dx,offset string mov ah,9 int 21h mov ax ,4c00h int 21h end start 〔习题 3.24〕 创建一个COM程序完成例题3.1的功能。 〔解答〕;lt301a,asm .model tiny .code .startup mov dx,offset string mov ah,9 int 21h .exit 0 string db 'Hello,Everybody!'0dh,0ah,'$' end ;〔习题 3.25〕 按下面要求写一个简化段定义格式的源程序 (1) 定义常量num,其值为5;数据段中定义字数组变量datalist,它的头5个字单元中 依次存放-1、0、2、5和4,最后1个单元初值不定; (2) 代码段中的程序将datalist中头num个数的累加和存入datalist的最后1个字单元中。 〔解答〕.model small .stack .data num equ 536 datalistdw -1,0,2,5,4,? .code .startup mov bx,offset datalist mov cx,num xor ax,axagain:add ax,[bx] inc bx inc bx loop again mov [bx],ax .exit 0 end〔习题 3.26〕 按下面要求写一个完整段定义格式的源程序 (1)数据段从双字边界开始,其中定义一个100字节的数组,同时该段还作为附加段; (2)堆栈段从节边界开始,组合类型为stack; (3)代码段的类别是‘code‘,指定段寄存器对应的逻辑段;主程序指定从100h开始, 给有关段寄存器赋初值;将数组元素全部设置为64h。 〔解答〕stack stack data array data code segment para ?stack? ends segment db 100 dup(?) ends segment ?code? org 100h start: mov ax,data mov ds,ax mov es,ax mov di,offset array mov al,64h mov cx,100 cld rep stosb mov ax,4c00h int 21h code ends dw 512 dup(?)assume cs:code,ds:data,es:data,ss:stack37 end start〔习题 3.27〕 编制程序完成两个已知双精度数 (4字节) A和B相加并将结果存入双精度变量单元SUM 中(不考虑溢出)。 〔解答〕;xt327.asm .model .stack .data A B sum dd h dd h dd ? .code .startup xor mov clc again: si, si cx, 2 ;相对于变量的位移量清零 ;分高低字分别相加,共两次 ;清零cf ;取第一个数的一个字(先低字后高字) ;取第二个数的一个字(先低字后高字) ;存和的一个字(先低字后高字) ;定义结果,执行后为:h ;定义两个双字的数(随意) small 256 ;定义堆栈段大小为256个字节mov ax, word ptr A[si] adc ax, word ptr B[si] mov word ptr sum[si], ax inc si inc si loop end again;修改位移量指向下一个字(加2) ;cx=cx-1 ,if cx&&0 ,jump again.exit 0〔习题 3.28〕 编制程序完成12H、45H、0F3H、6AH、20H、0FEH、90H、0C8H、57H和34H等10个 字节数据之和,并将结果存入字节变量SUM中(不考虑溢出)。 〔解答〕.startup xor si, si ;位移量清零 ;取第一个数 ;累加次数 ;指向下一个数 ;累加 ;如未完,继续累加 ;完了,存结果 mov al, bdata[si] mov cx, num-1 again: inc si adc al, bdata[si] loop again mov sum, al .exit 0 end38 〔习题 3.29〕 结构数据类型如何说明、结构变量如何定义、结构字段如何引用? 〔解答〕 结构类型的说明使用一对伪指令STRUCT(MASM5.x是STRUC,功能相同)和ENDS。 它们的格式为: 结构名 STRUCT ? ;数据定义语句 结构名 ENDS 结构变量定义的格式为: 变量名,结构名 〈字段初值表〉 引用结构字段,采用圆点―.‖操作符,其格式是: 结构变量名.结构字段名。 〔习题 3.30〕 记录数据类型如何说明,记录变量如何定义,width和mask操作符是什么作用? 〔解答〕 记录类型的说明采用伪指令RECORD,它的格式为: 记录名 RECORD 位段[,位段?] 定义记录变量的格式: 记录变量名 记录名 〈段初值表〉 Width记录名/记录位段名操作符返回记录或记录位段所占用的位数。 mask记录位段名操作符返回一个8位或16位数值,其中对应该位段的个位为1,其余位 为0。第 4 章 基本汇编语言程序设计〔习题 4.1〕 例题4.2如果要求算术右移8位,如何修改程序。 〔解答〕 思路: 首先由最高位字节向次低位字节传送??次低位字节向最低位字节传送(共7 次);再判最高位字节符号位,如为0,送00h到最高位字节;如为1,送ffh到最高位字节。 传送可参考例题4.2,不过应从第一号字节送第零号字节,??最高位字节向次低位字节传 送;也可以用循环来完成: .model small .stack 256 .data dq 4321h .code .startup mov cx,7 mov si,1 mov al, byte ptr qvar[si] mov byte ptr qvar[si-1],al39qvaragain: ezz: done:inc si loop again test al,80h jz ezz mov bl,0ffh jmp done mov bl,0 mov byte ptr qvar[7],bl .exit 0 end〔习题 4.2〕 例题4.2如果要求算术左移7位,如何用移位指令实现。 〔解答〕 思路:可设计外循环体为8个字节左移一次,方法是:最低位字节算术左移一次, 次低位字节至最高位字节依次带 CF 位循环左移一次(内循环共8次),外循环体控制执行 7次即可。 .model small .stack 256 .data dq 4321h .code .startup mov dx, 7 ;外循环次数 mov ax, byte ptr qvar[0] ;最低位字节送ax shl ax, 1 ;最低位字节左移一次,其d7移入CF位 mov si, 1 mov cx, 7 rcl byte ptr qvar[si], 1 inc si loop again dec dx jnz lpp .exit 0 .end 〔习题 4.3〕 将AX寄存器中的16位数连续4位分成一组,共4组,然后把这4组数分别放在AL、BL、 CL和DL寄存器中。 〔解答〕 思路:设这四组从低位到高位分别放在AL、BL、CL和DL寄存器中。仅列出代码段: ;内循环次数 ;高位字节依次左移 P50qvarlpp:again:40 mov bl, al and al, 0fh mov cl, 4 shr bl, cl mov dl, ah and dl, 0f0h mov cl, 4 shr dl, cl mov cl, ah and cl, 0fh;将al中的两组分开 ;屏蔽高四位后送al ;原al中的数据逻辑右移4次送bl ;将ah中的两组分开 ;屏蔽低高四位后送dl ;原ah中的数据逻辑右移4次送dl ;屏蔽高四位后送cl〔习题 4.4〕 编写一个程序,把从键盘输入的一个小写字母用大写字母显示出来。 〔解答〕getkey: mov ah, 1 int 21h cmp al, 'a' jb cmp ja getkay al, 'z' getkay ;是小写字母转换为大写字母 ;显示 ;判键值是小写字母? ;从键盘输入,出口:al存键值sub al,20h mov dl,al mov ah, 02h int 21h〔习题 4.5〕 已知用于LED数码管显示的代码表为:LEDtable DB 0c0h,0f9h,0a4h,0b0h,99h,92h,82h,0f8h DB 80h,90h,88h,83h,0c6h,0c1h,86h,8eh它依次表示0 ~ 9、A ~ F这16个数码的显示代码。现编写一个程序实现将lednum中的一 个数字(0 ~ 9、A ~ F)转换成对应的LED显示代码。 〔解答〕.model small .stack 256 .data LEDtable lednum DB 0c0h,0f9h,0a4h,0b0h,99h,92h,82h,0f8h DB 80h,90h,88h,83h,0c6h,0clh、86h,8eh DB ? .code .startup mov bx, offset LEDtable mov al, lednum41 mov ax,Xsub ax,Y N OF=1Y overflow结果正确xlat .exit 0 end;al中存有对应的LED显示代码〔习题 4.6〕 请问第2章例题2.39 / 2.41 / 2.43 / 2.46的分支是单分支、双分支或多分支结构? 〔解答〕 例题2.43/2.45/2.47 为单分支结构;2.50为多分支结构。 〔习题 4.7〕 分析第2章例题2.40的分支结构,画出流程图。 〔解答〕 〔习题4.8〕 如果在例题4.4的table中依次填入msg1 ~ msg8,程序应该怎样修改? 〔解答〕 (1) 将jmp table[bx]指令改为:mov dx, table[bx] (2) 去掉源程序中:.exit 0---end之间的语句 〔习题 4.9〕 编制一个程序,把变量bufX和bufY中较大者存入bufZ;若两者相等,则把其中之一存 入bufZ中。假设变量存放的是8位无符号数。 〔解答〕.model small .stack 256 .data bufx bufY bufz db ? db ? db ? .code .startup mov al, bufX mov bl, bufY cmp al,bl ja next mov bufZ, bl jmp done 42 next: mov bufZ, al done: .exit 0 end〔习题 4.10〕 设变量bufX为有符号16位数,请将它的符号状态保存在signX,即:如果X大于等于0, 保存0;如果X小于0,保存-1(ffh)。编写该程序。 〔解答〕.model small .stack .data bufX signX dw -7 db ? .code .startup cmp bufX,0 jl next mov signX,0 jmp done next: done: mov signX,-1 .exit 0test bufX,80jnz next〔习题 4.11〕 bufX、bufY和bufZ是3个有符号16进制数,编写一个比较相等关系的程序: (1)如果这3个数都不相等,则显示0; (2)如果这3个数中有两个数相等,则显示1; (3)如果这3个数都相等,则显示2。 〔解答〕.model small .stack 256 .data bufx bufY bufz dw ? dw ? dw ? .code .startup mov ax, mov bx, mov cx, bufX bufY bufZmov dl, '0' cmp ax,bx43 jnz next1 inc dl next1: cmp ax,cx jnz next2 inc dl next2: cmp bx,cx jnz next3 inc dl next3: cmp dl,?3? jb next4 mov dl,?2? next4: mov ah,02h int 21h .exit 0 end ;显示〔习题 4.12〕 分析第2章例题2.42、2.52、2.53的结构,分别画出它们的流程图。 〔解答〕 2.42 xor al,albx=0 Y shl bx,1 Y CF=0 inc al next 2.52 mov si , offset source mov di,offset destination mov cx, 10044 cld movsb dec cx jnz2.53ex=0mov ax, 0mov di. 0mov ax, 8000hcldstoswdec cx jnz cx=0 〔习题 4.13〕 例题4.8内外循环次数共是多少?如果要求按从大到小排序,程序如何修改? 〔解答〕 外循环次数是: count-1次 (19次) 内循环次数是: (count-1)!次 ( 19! 次) 内外循环次数共是 count-1 + (count-1)!次 ,即 19+19!次 〔习题 4.14〕 串操作指令常要利用循环结构, 现在不用串操作指令实现字符串string1内容传送到字符 串string2,字符长度为count。 〔解答〕.model small .stack 256 .data string1 db 'good morning!' 45 len string2= $-string1 db len dup(?) .code .startup mov cx, len mov si, offset string1 mov di, offset string2 ;字符数 ;源指针 ;目标指针 ;送一个字符 ;修改指针 ;修改指针 ;cx=cx-1,cx=0时退出循环again:mov al, [si] mov [di], al inc si inc di loop again .exit 0 end〔习题 4.15〕 不用串操作指令求主存0040h : 0开始的一个64KB物理段中共有多少个空格? 〔解答〕 这里仅列出主程序段:mov ax,0040h mov ds, ax xor si, si xor cx, cx xor dx, dx again: cmp [si], 20h jne next: inc si loop again .exit 0 end next inc dx ;偏移量地址 ;计数(循环次数) ;空格计数器清零 ;与空格的ASCII码比较 ;不是空格,转 ;是空格,空格数加1 ;修改地址指针 ;cx=cx-1,如cx=0 退出循环 ;送段地址〔习题 4.16〕 编程实现把键入的一个字符,用二进制形式(0 / 1)显示出它的ASCII代码值。 〔解答〕.model small .stack 256 .data stri1 db 'please input',0dh,0ah,'$' .code .startup mov dx,offset stri146 mov ah,09h int 21h mov ah, 01h int 21h mov cx, 8 again: xor dl,dl shl al, 1 adc dl, '0' mov ah,02h int 21h loop again .exit 0 end〔习题 4.17〕 编写程序,要求从键盘接收一个数bellN(0 ~ 9),然后响铃bellN次。 〔解答〕.model small .stack .data stri1 db 'please input number:1--9',0dh,0ah,'$' .code .startup again: mov dx,offset stri1 ;显示stri1,提示输入 mov ah,09h int 21h mov ah,01h int jb ja 21h again again ;在?1?--?9?,屏蔽高4位 ;振铃次数送cx ;调用一次振铃 cmp al, '1' cmp al, '9' and al,0fh mov cl, al xor ch,ch abc: mov dl, 07h mov ah, 02h int 21h loop abc .exit 0 end ;调用输入一个字符 ;输入一个字符存在al中 ;判该字符,如不在?1?--?9? ;重新输入47 〔习题 4.18〕 编写程序,将一个包含有20个有符号数据的数组arrayM分成两个数组:正数数组arrayP 和负数数组arrayN,并分别把这两个数组中的数据个数显示出来。 〔解答〕.model small include io.inc .stack .data arrayM arrayP arrayN dispP dispN db db 1,2,3,4,5,6,0,-3,-5,-6,0,7,6,90,-18,-23,34,7,9,8 ;源数组 20 dup(?) ;正数数组 ;负数数组db 20 dup(?) db 'Plus Number: ','$'db 0dh,0ah,'Negs Number: ','$' .code .startup mov cx, 20 xor bx, bx xor si, si ;源数组元素数 ;设bh为正数个数,bl为负数个数,均清零 ;源数组地址位移量 ;循环20次 ;取一个元素 ;判正数 ;是,正数个数加1 ;否,负数个数加1 ;修改位移量 ;循环次数减1 ;调用显示字符串功能 ;显示个数again: mov ax, arrayM[si] cmp ax,0 jl Neg inc bh jmp next Neg: next: inc bl inc si loop again mov ah,09h mov dx, offset dispP int 21h mov al,bh call dispuib mov ah,09h mov dx, offset dispN int 21h mov al,bl call dispuib .exit 0 end; 调用I/O子程序库中的子程序 ;调用显示字符串功能 ;显示个数; 调用I/O子程序库中的子程序48 〔习题 4.19〕 编写计算100个正整数之和的程序。如果和不超过16位字的范围(65535),则保存其 和到wordsum,如超过则显示?overflow‘。 〔解答〕.model small .stack .data num wlist wordsum error equ 100 dw num dup(?) dw ? db 'overflow. $' .code .startup mov bx,offset wlist mov cx,num xor ax,ax again: add ax,[bx] jc next inc bx inc bx loop again mov [bx],ax jmp done next: mov dx,offset error mov ah,9 int 21h done: .exit 0 end〔习题 4.20〕 编程判断主存0070h:0开始的1KB中有无字符串?DEBUG‘。这是一个字符串包含的问 题,可以采用逐个向后比较的简单算法。 〔解答〕.model small .stack .data disp1 disp2 db 'There is DEBUG in the aera!' ,0dh,0ah,?$' db 'There is no DEBUG in the aera!' ,0dh,0ah,?$' .code .startup mov ax, 0070h ;送段地址49 mov ds, ax xor si, si mov cx,1024 cmp [si], 'D' jne next inc si cmp [si], 'E' jne next inc si cmp [si], 'B' jne next inc si cmp [si], 'U' jne next inc si cmp [si], 'G' je yes next: no: yes: dsp: inc si loop again mov dx, offset disp2 jmp dsp ;找到,显示disp1 mov dx, offset disp1 mov ah, 09h int 21h .exit 0 end ;是?DEBUG',转 ;不是,地址增1 ;循环 ;没找到,显示disp2 ;与?D'比较 ;不是,转 ;是,地址增1 ;同上 ;地址指针清零〔习题 4.21〕 编程把一个16位无符号二进制数转换成为用8421BCD码表示的5位十进制数。转换算法 可以是:用二进制数除以10000,商为―万位‖,再用余数除以1000,得到―千位‖;依次用余 数除以100、10和1,得到―百位‖、―十位‖和―个位‖。 〔解答〕.model small .stack 256 .data var dbcd dw db 3546 5 dup(?).code .startup mov ax, var mov bx, 1000050 mov cl, 10 xor si, si xor dx, dx again: div inc div bx si cl mov dbcd[si], al xchg ax, bx xchg ax, bx cmp si,5 jnz again .exit 0 end〔习题 4.22〕 过程定义的一般格式是怎样的?子程序入口为什么常有PUSH指令、出口为什么有POP 指令?下面的程序段有什么不妥吗?若有,请改正:crazy PROC push ax xor ax,ax xor dx,dx again: add ax,[bx] adc dx,0 inc bx inc bx loop again ret ENDP crazy〔解答〕crazy PROC push ax xor ax,ax xor dx,dx again: add ax,[bx] adc dx,0 inc bx inc bx loop again ret ENDP crazy ;crazy PROC ; ; ; ; ; ; ; ; xor ax,ax xor dx,dx add ax,[bx] adc dx,0 inc bx inc bx loop again ret ENDP;again:; crazy51 〔习题 4.23〕 子程序的参数传递有哪些方法,请简单比较。 〔解答〕 寄存器、共享变量(公共存储单元)、堆栈 用寄存器传递参数是把参数存于约定的寄存器中,这种方法简单易行,经常采用; 用变量传递参数是主程序与被调用过程直接用同一个变量名访问传递的参数,就是利 用变量传递参数。如果调用程序与被调用程序在同一个源程序文件中,只要设置好数据段 寄存器DS,则子程序与主程序访问变量的形式相同,也就是它们共享数据段的变量,调用 程序与被调用程序不在同一个源文件中,必须利用public/extern进行声明,才能用变量传递 参数,利用变量传递参数,过程的通用性比较差,然而,在多个程序段间,尤其在不同程 序的模块间,利用全局变量共享数据也是一种常见的参数传递方法; 用堆栈传递参数是主程序将子程序的入口参数压入堆栈,子程序从堆栈中取出参数; 子程序将出口压入堆栈,主程序弹出堆栈取得它们。 〔习题 4.24〕 采用堆栈传递参数的一般方法是什么,为什么应该特别注意堆栈平衡问题。 〔解答〕 采用堆栈传递参数的一般方法是主程序将子程序的入口参数压入堆栈,子程序从堆栈 中取出参数子程序将出口参数压入堆栈,主程序弹出堆栈取得它们。因为堆栈采用“先进 后出”原则存取,而且返回地址和保护的寄存器等也要存于堆栈,所以要特别注意堆栈平 衡问题。 〔习题 4.25〕 什么是子程序的嵌套、递归和重入? 〔解答〕 子程序中又调用子程序就形成子程序嵌套。 子程序中直接或间接调用该子程序本身就形成子程序递归。 子程序的重入是指子程序被中断后又被中断服务程序所调用,能够重入的子程序称为 可重入子程序。 〔习题 4.26〕 将例题4.7的大写转换为小写字母写成过程,利用AL作为入口、出口参数完成。 〔解答〕 .model small .stack 256 .data stdng db 'HeLLO eveRyboDy ! ' , 0 .code .startup mov bx, offset atring again: mov al, [bx] call chan ;调用过程 mov [bx] , al next: inc bx52 jmp again done: .exit 0 chan proc or al, al jz done cmp al,'A' jb next cmp al, 'Z' ja next or al, 20h ret chan endp end;大写转换为小写字母的过程〔习题 4.27〕 请按如下子程序说明编写过程:;子程序功能:把用ASCII码表示的两位十进制数转换为对应二进制数 ;入口参数:DH=十位数的ASCII码,DL=个位数的ASCII码 ;出口参数:AL=对应的二进制数〔解答〕astob proc and dh, 0fh mov al, dh mul 10 and dl, 0fh add al, dl ret astob endp ;al= 10*dh ;个位数的ASCII码转为二进制数 ;al= 10*dh + dl ;十位数的ASCII码转为二进制数〔习题 4.28〕 写一个子程序,根据入口参数AL=0/1/2,分别实现对大写字母转换成小写、小写转换 成大写或大小写字母互换。欲转换的字符串在string中,用0表示结束。 〔解答〕Change proc Push bx xor jz jz bx, bx chan_0 dec al chan_1 dec al cmp al,0 ;保护bx ;位移量清零 ;根据入口参数AL=0/1/2,分别处理53 jz chan_0:chan_2 jmp done mov al,string[bx] cmp al,0 jz done cmp al,?A? jb next0 cmp al,?Z? ja next0 add al, 20h mov string[bx], al ;转换 ;位移量加1,指向下一字母 ;实现对小写字母转换成大写 ;是大写字母 ;是大写字母 ;实现对大写字母转换成小写next0: chan_1:inc bx jmp chan_0 mov al,string[bx] cmp al,0 jz done cmp al,?a? jb next1 cmp al,?z? ja next1 sub al, 20h mov string[bx], al;是大写字母 ;是大写字母 ;转换 ;位移量加1,指向下一字母 ;实现对大写字母转换成小写next0: chan_2:inc bx jmp chan_1 mov al,string[bx] cmp al,0 jz done cmp al,?A? jb next2 cmp al,?Z? ja next20 add al, 20h jmp next2;是大写字母 ;是大写字母 ;转换 ;是大写字母 ;是大写字母 ;转换 ;位移量加1,指向下一字母next20:cmp al,?a? jb next2 cmp al,?z? ja next2 sub al, 20hmov string[bx], al next2: inc bx54 jmp chan_2 done: change pop bx ret endp ;恢复bx〔习题 4.29〕 编制一个子程序把一个16位二进制数用十六进制形式在屏幕上显示出来,分别运用如 下3种参数传递方法,并用一个主程序验证它。 (1)采用AX寄存器传递这个16位二进制数 (2)采用wordTEMP变量传递这个16位二进制数 (3)采用堆栈方法传递这个16位二进制数 〔解答〕 (1)采用AX寄存器传递这个16位二进制数.model small .stack .data wdata dw 34abh .code .startup mov ax,wdata call dispa .exit 0 ; dispa proc push cx push dx mov cl,4 mov dl,ah shr dl,cl call dldisp mov dl,ah and dl,0fh call dldisp mov dl,al shr dl,cl call dldisp mov dl,al and dl,0fh call dldisp pop dx pop cx55 re proc push ax or dl,30h cmp dl,39h jbe dldisp1 add dl,7 dldisp1: mov ah,2 int 21h pop ax ret dldisp endp end(2)采用wordTEMP变量传递这个16位二进制数.model small .stack .data wdata wordtemp dw 34abh dw ? .code .startup mov ax,wdata mov wordtemp,ax call dispa .exit 0 ; dispa proc push cx push dx mov cl,4 mov dl,byte ptr wordtemp+1 shr dl,cl call dldisp mov dl,byte ptr wordtemp+1 and dl,0fh call dldisp mov dl,byte ptr wordtemp shr dl,cl56 call dldisp mov dl,byte ptr wordtemp and dl,0fh call dldisp pop dx pop cx re proc push ax or dl,30h cmp dl,39h jbe dldisp1 add dl,7 dldisp1: mov ah,2 int 21h pop ax ret dldisp endp end(3)采用堆栈方法传递这个16位二进制数.model small .stack .data wdata dw 34abh .code .startup push wdata call dispa pop ax .exit 0 ; dispa proc push bp mov bp,sp push ax push cx push dx mov ax,[bp+4] ;add sp,257 mov cl,4 mov dl,ah shr dl,cl call dldisp mov dl,ah and dl,0fh call dldisp mov dl,al shr dl,cl call dldisp mov dl,al and dl,0fh call dldisp pop dx pop cx pop ax pop bp re proc push ax or dl,30h cmp dl,39h jbe dldisp1 add dl,7 dldisp1: mov ah,2 int 21h pop ax ret dldisp endp end〔习题 4.30〕 设有一个数组存放学生的成绩(0 ~ 100),编制一个子程序统计0 ~ 59分、60 ~ 69分、 70 ~ 79分、80 ~ 89分、90 ~ 100分的人数,并分别存放到scoreE、scoreD、score C、score B 及score A单元中。编写一个主程序与之配合使用。 〔解答〕.model small .stack .data58 score count scoreE scoreD scoreC scoreB scoreAdb 70,86,90,45,60,96,100,0,... equ $-score db ? db ? db ? db ? db ? .code .startup lea bx, score mov cx, count ;总人数;全班成绩数组;0--59分人数 ;60--69分人数 ;70--79分人数 ;80--89分人数 ;90--99分人数again: mov al,[bx] call tjrs inc bx loog again .exit 0 tjrs proc cmp al, 60 jae next0 ;al&= 60转 ;al&60,0--59分的人数加1 inc scoreE jmp next4 next0: cmp al, 70 jae inc next1 scoreD ;统计分段人数 ;取一个成绩 ;调用统计分段人数 ;调整指针 ;cx-cx-1,cx=0退出循环jmp next4 next1: cmp al, 80 jae inc next2 scoreCjmp next4 next2: cmp al, 90 jae inc jmp next3: next4: tjrs inc ret endp end next3 scoreB next4 scoreA〔习题 4.31〕 编写一递归子程序,计算指数函数X n 的值。59 〔解答〕 model small .stack .data x dw 5 n dw 6 zish dw ? .code .startup mov bx,x push bx mov bx,n push bx mov bx,x call zshhsu pop zish add sp,2 .exit 0 zshhsu proc push bx push ax push bp mov bp,sp mov ax,[bp+8] cmp ax,0 jne zshhsu1 inc ax jmp zshhsu2 zshhsu1: dec ax push bx push ax call zshhsu pop ax pop bx mul bx zshhsu2: mov [bp+8],ax pop bp pop ax pop bx60 ret zshhsu endp end第5章高级汇编语言程序设计〔习题 5.1〕 条件表达式中逻辑与―&&‖表示两者都为真,整个条件才为真,对于程序段:.if (X==5) && (ax!=bx) inc ax .endif请用转移指令实现上述分支结构,并比较汇编程序生成的代码序列。 〔解答〕cmp X, 5 je jmp abc: jne jmp cde: done: abc done cde donecmp ax, bxinc ax ....〔习题 5.2〕 条件表达式中逻辑与―||‖表示两者之一为真,整个条件就为真,对于程序段:.if (X==5) || (ax!=bx) inc ax .endif请用转移指令实现上述分支结构,并比较汇编程序生成的代码序列。 〔解答〕cmp X, 5 je jmp jne jmp abc: abc done abc donecmp ax, bxinc axdone: ....〔习题 5.3〕 对于程序段:.while ax!=10 mov [bx],ax61 inc bx inc bx inc ax .endw?请用处理器指令实现上述循环结构,并比较汇编程序生成的代码序列。 〔解答〕 again: cmp ax, 10 ;相当于.while 语句 .while ax!=10 je done ;同上 mov [bx],ax mov [bx],ax ;循环体不变 inc bx inc bx inc bx inc bx inc ax inc ax .endw jmp again ;相当于.endw 语句 done: ... 〔习题 5.4〕 对于程序段:.repeat mov [bx],ax inc bx inc bx inc ax .until ax==10?请用处理器指令实现上述循环结构,并比较汇编程序生成的代码序列。 〔解答〕 again: nop ;相当于.repeat .repeat mov [bx], ax ;循环体不变 mov [bx],ax inc bx inc bx inc bx inc bx inc ax inc ax cmp ax,10 ;相当于.until 语句 .until ax==10 jne again ;同上 〔习题 5.5〕 宏是如何定义、调用和展开的? 〔解答〕 宏定义采用一对伪指令实现,格式如下(其中方括号表示可选): 宏名 macro [形参表] 宏定义体 endm 宏调用时,利用宏名带上实参即可,格式如下: 宏名 [实参表] 〔习题 5.6〕 宏定义中的形式参数有什么特点,它是如何进行形参和实参结合的? 〔解答〕62 宏的参数功能非常强大,即可以没有参数,也可以带有一个或多个参数;而且参数的 形式非常灵活,可以是常量、变量、存储单元、指令或它们的一部分,还可以是表达式等。 宏展开时的形参与实参结合是用位置匹配的实参对形参进行取代。实参个数与形参个 数不同时,多余的实参没有作用,缺少的实参则用空格代替。取代时,不进行有效性等检 查。 〔习题 5.7〕 宏结构和子程序在应用中有什么不同,如何选择采用何种结构? 〔解答〕 宏汇编的特点是在汇编阶段进行语句展开,无需返回,不减少目标程序代码,执行速 度没有改变;通过形参、实参结合传递参数。宏汇编适合于程序段较短或要求执行速度快 的场合。 子程序在执行阶段需要由主程序执行调用指令CALL进行调用,进而转入子程序执行, 子程序执行结束执行返回指令RET返回主程序。 子程序如果被多次使用, 则会减少目标程序 代码;主程序与子程序之间主要采用寄存器、存储单元、堆栈等传递参数。子程序适合于 程序段较长或为减少目标代码的场合。 〔习题 5.8〕 宏汇编、重复汇编与条件汇编有什么异同? 〔解答〕 在汇编过程中实现宏展开的汇编称为宏汇编,实现程序过程中连续地重复一段相同或 者基本相同的语句的汇编是重复汇编;根据某种条件确定是否汇编某段语句序列的汇编是 条件汇编 〔习题 5.9〕 对于例题5.5b的宏定义shrot,对应如下宏指令的宏展开是什么? shrot word ptr [bx],4,ror 〔解答〕 shrot word ptr [bx],4,ror 1 push cx 1 mov cl,4 1 ror word ptr [bx],cl 1 pop cx 〔习题 5.10〕 定义一个宏logical,用它代表4条逻辑运算指令:and/or/xor/test,注意需要利用3个形式 参数,并给一个宏调用以及对应宏展开的例子。 〔解答〕 macro lcode,dopd,sopd lcode dopd,sopd endm 例如,如果使用―and ax,[bx]‖指令,可以利用该宏定义,写出宏指令如下: logical and,ax,[bx] 初看起来,这个问题似乎比较难解决。实际上,这4条逻辑指令都具有相同的指令操作 数格式,所以可以归纳为如上宏定义。63logical 〔习题 5.11〕 必要时做一点修改,使在上一习题定义的宏logical能够把neg指令包括进去,也请给一 个使用neg指令的宏调用以及对应宏展开的例子。 〔解答〕 宏定义:logical Micro srop1,srop2,scode1,scode2 mov ax,srop1 mov bx,srop2 scode1 ax,bx scode2 ax mov scop1,ax endm宏调用:logical wordX,wordY,and,neg宏展开:1 1 1 1 1 mov ax, wordX mov bx, wordY and ax,bx neg ax mov wordX,ax〔习题 5.12〕 编写一个宏指令move doprnd,soprnd,它实现任意寻址方式的字量源操作数送到目的操 作数,包括存储单元到存储单元的传送功能。 〔解答〕 move MACRO doprnd,soprnd push ax mov ax,soprnd mov doprnd,ax pop ax ENDM 〔习题 5.13〕 定义一个宏movestr strN,dstr,sstr, 它将strN个字符从一个字符区sstr传送到另一个字符区 dstr。 〔解答〕movstr strN,dstr,sstr lea si, sstr lea di, dstr mov cx, strN cld rep movsb64 endm〔习题 5.14〕 给出宏定义如下: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试展开以下宏调用:① ABSDIF P1,P2,DISTANCE ② ABSDIF [BX],[SI],[DI]〔解答〕①ABSDIF Pl,P2,DISTANCE 1 push ax 1 mov ax,P1 1 sub ax,P2 1 cmp ax,0 1 jge ?? 0000 1 neg ax 1 ?? 0000 1 mov DISTANCE, ax 1 pop ax ②ABSDIF [BX],[SI],[DI] 1 push ax 1 mov ax,[bx] 1 sub ax,[si] 1 1 1 cmp ax,0 neg ax ?? 0000 1 mov [di],ax 1 pop ax 1 jge ?? 0000〔习题 5.15〕 利用重复汇编方法定义一个数据区,数据区有100个双字,每个双字的高字部分依次是 2、4、6、?、200,低字部分都是0。 〔解答〕 wtemp =2 repeat 100 dw 0 dw wtemp =wtemp+265wtemp endm 〔习题 5.16〕 利用宏结构完成以下功能:如果名为byteX的数据大于5时,指令―ADD AX,AX‖将汇编 10次,否则什么也不汇编。 〔解答〕 finsum macro num1,num2,sum if num1 gt num2 mov ax,num1 add ax,num2 else mov ax,num1 shl ax,1 add ax,num2 endif mov sum,ax endm .model small .stack .data =3234h =2345h =4567h dw ? dw ? .code .startup finsum wordX1,wordY1,sum1 finsum wordX1,wordY2,sum2 .exit 0 endwordX1 wordY1 wordY2 sum1 sum2〔习题 5.17〕 用宏结构实现宏指令FINSUM,它比较两个数wordX和wordY,若wordX & wordY,则 执行SUM←wordX+2×wordY,否则执行SUM←2×wordX+wordY。 〔解答〕 finsum macro num1,num2,sum if num1 gt num2 mov ax,num1 add ax,num2 else66 wordX1 wordY1 wordY2 sum1 sum2mov ax,num1 shl ax,1 add ax,num2 endif mov sum,ax endm .model small .stack .data =3234h =2345h =4567h dw ? dw ? .code .startup finsum wordX1,wordY1,sum1 finsum wordX1,wordY2,sum2 .exit 0 end〔习题 5.18〕 DOS功能调用采用AH存放子功能号,而有些功能需要DX存放一个值。定义一个宏 DOS21H,实现调用功能;如果没有提供给DX的参数,则不汇编给它赋值的语句。 〔解答〕 DOS21H MACRO callnum,calladdress mov ah,callnum ifnb &calladdress& mov dx,calladdressendifint 21h ENDM 〔习题 5.19〕 将例题4.7的大写转换为小写字母写成宏完成。 〔解答〕 dxgxx macro buf local again,next1,next2 mov bx,offset buf mov al,[bx] or al,al67again: jz next2 cmp al,'A' jb next1 cmp al,'Z' ja next1 or al,20h mov [bx],al next1: inc bx jmp again next2: endm .model small .stack .data db 'HeLLO,eveRyboDy !',0 .code .startup dxgxx string mov bx,offset string mov dl,[bx] or dl,dl jz done mov ah,2 int 21h inc bx jmp w1 .exit 0 endstringw1:done:〔习题 5.20〕 请用宏结构实现例题4.4数据段中8个标号组成的地址表, 以及代码段的8个处理程序段; 然后运行通过。 〔解答〕 .model small .stack .data db 'input number(1~8):',0dh,0ah,'$' db 'chapter 1:fundamentals of assembly language',0dh,0ah,'$' db 'chapter 2:8086 instruction set',0dh,0ah,'$'68msg msg1 msg2 msg3 msg4 msg5 msg6 msg7 msg8 table FORC ENDMdb 'chapter 3:statements of assembly language',0dh,0ah,'$' db 'chapter 4:basic assembly language programming',0dh,0ah,'$' db 'chapter 5:advanced assembly language programming',0dh,0ah,'$' db 'chapter 6:32-bit instructions and programming',0dh,0ah,'$' db 'chapter 7:mixed programming with c/c++',0dh,0ah,'$' db 'chapter 8:fp instructions and programming',0dh,0ah,'$' equ this byte num1, dw disp&num1&.code .startup s}

我要回帖

更多关于 汇编语言二进制输出 的文章

更多推荐

版权声明:文章内容来源于网络,版权归原作者所有,如有侵权请点击这里与我们联系,我们将及时删除。

点击添加站长微信