IBM-PC汇编语言程序设计答案 下载本文

49703100.doc - 36 - 作者:User

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

SING4: ┇

JMP BEGIN

SING5: ┇

JMP BEGIN

EXIT: RET MAIN ENDP CSEG ENDS ;以上定义代码段 ;--------------------------------------------------------------------------

END START

5.25 试用8086的乘法指令编制一个32位数和16位数相乘的程序;再用80386的乘法指令编制一个32

位数和16位数相乘的程序,并定性比较两个程序的效率。 答:8086的程序如下(假设为无符号数):

DSEG SEGMENT MUL1 DD ? ;32位被乘数 MUL2 DW ? ;16位乘数 MUL0 DW 0,0 ,0 ,0 ;乘积用64位单元存放 DSEG ENDS

49703100.doc - 37 - 作者:User

;-------------------------------------------------------------------------- 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

START: PUSH DS ;设置返回DOS

SUB AX, AX PUSH AX

MOV AX, DSEG MOV DS, AX ;给DS赋值

BEGIN: 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.26 如数据段中在首地址为MESS1的数据区内存放着一个长度为35的字符串,要求把它们传送到附

加段中的缓冲区MESS2中去。为提高程序执行效率,希望主要采用MOVSD指令来实现。试编写这一程序。

49703100.doc - 38 - 作者:User

答: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.27 试用比例变址寻址方式编写一386程序,要求把两个64位整数相加并保存结果。

答:80386的程序如下:

.386

.MODEL SMALL .STACK 100H .DATA

DATA1 DQ ? DATA2 DQ ?

.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

答:程序有错。改正如下:

49703100.doc - 39 - 作者:User

CRAY CRAY

PROC ADD RET ENDP

AX, BX

;CRAY是过程名,应放在ENDP的前面

6.2 已知堆栈寄存器SS的内容是0F0A0H,堆栈指示器SP的内容是00B0H,先执行两条把8057H和

0F79BH分别入栈的PUSH指令,然后执行一条POP指令。试画出示意图说明堆栈及SP内容的变化过程。

9BH 再PUSH,SP-2 答:变化过程如右图所示:

F7H POP, SP+2 6.3 分析下面的程序,画出堆栈最满时各单元的地址及内容。先 PUSH,SP-2 57H 80H ;********************************************

F0A0:00B0H S_SEG SEGMENT AT 1000H ;定义堆栈段

6.2 题堆栈及SP内容的变化过程 DW 200 DUP (?) ;200*2=190H

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) PUSH DS :018A T_ADDR MOV AX, 0 :018C 0000 PUSH AX :018E (DS) ┇ PUSH T_ADDR SP: 0186 PUSH AX

6.3 题堆栈最满时各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