汇编语言程序设计(第二版) 钱晓捷 习题答案
第二章(01)
2.1
(1)AX=1200h
(2)AX=0100h
(3)AX=4C2Ah
(4)AX=3412h
(5)AX=4C2Ah
(6)AX=7856h
(7)AX=65B7h
IP 指令指针禁止用户访问
;获取 table 的首地址,BX=200H
;传送欲转换的数字,AL=8
;转换为格雷码,AL=12H
立即数不允许传给段寄存器
段寄存器之间不允许传送
两操作数类型不匹配
目的操作数应为[ BP ]
源操作数应为 [BX+DI]
立即数不能作目的操作数
2.2(1) 两操作数类型不匹配
(2)
(3)
(4)
(5)
(6)
(7)
(8)
2.3
lea bx,table
mov al,8
xlat
2.4
堆栈是一种按“先进后出”原则存取数据的存储区域。
堆栈的两种基本操作是压栈和出栈,对应的指令是 PUSH 和 POP。
2.5
mov ax,8057h
push ax
mov ax,0f79h
push ax
pop bx
pop [bx]
2.6
AL=89h
AL=12h
AL=0afh
AL=0afh
AL=00h
AL=0ffh
0
;bx=0f79h
;DS:[0f79h]=8057h
SF
0
1
1
0
1
OF
1
0
0
0
0
PF
1
1
1
1
CF
1
0
1
0
0
ZF
0
0
0
1
0
0
1
2.7
W=X+Y+24-Z
1 AL=00h
0
1
2.8
(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 字操作数改为 A,B,C,D
;ax=A
;dx,ax = A*B
<--
(将操作数看作符号数,以下同)
;bx,ax
;ax = C
;dx,ax =C (扩展符号后为双字)
dx,ax =A*B
mov ax,X
imul Y
mov cx,ax
mov bx,dx
mov ax,Z
cwd
add cx,ax
adc bx,dx
sub cx,540
sbb bx,0
mov ax, V
cwd
sub ax, cx
sbb dx, bx
idiv X ;运算结果:[D-(A*B+C-540h)]/A ;ax 存商,dx 存余数
;bx,cx<-- A*B+C-540
;ax= D
;dx,ax= D (扩展符号后为双字)
;dx,ax = dx,ax - bx,cx = D-(A*B+C-540)
;bx,cx <-- bx,cx+dx,ax=A*B+C
2.10;(1)xchg 的操作数不能是立即数
(2 不能对 CS 直接赋值
(3)两个操作数不能都是存储单元
(4)堆栈的操作数不能是字节量
(5)adc 的操作数不能是段寄存器
(6)没有确定是字节还是字操作
(7)in 不支持超过 FFH 的直接寻址
(8)out 只能以 AL/AX 为源操作数
第二章(02)
2.11;
指令
Mov
And
Or
Xor
Not
Test
注意: 1. mov, not 指令不影响标志位
AX 的值 CF OF SF ZF PF
1470h
ax,1407h
0
1470h
ax,ax
0
1470h
ax,ax
0
0
0
ax,ax
-
ax 0ffffh
-
0ffffh
ax,0f0f0h
-
0
0
1
-
0
-
0
0
0
-
0
-
0
0
1
-
1
-
0
0
0
-
1
2. 其他逻辑指令使 CF=OF=0, 根据结果影响其他标志位。
第 2 章
(03)
2.12;
指令
mov si,ax
shl si,1
add si,ax
mov dx,bx
mov cl,03h
shl dx,cl
sub dx,bx
add dx,si
注释
si=ax
si=2*ax
si=3*ax
dx=bx
cl=03h
dx=8*bx
dx=7*bx
dx=7*bx+3*ax
执行结果
si=0008h
si=0010h
si=0018h
dx=0010h
dx=0080h
dx=0070h
dx=0088h
CF OF SF ZF PF
-
-
0
0
-
0
0
0
0
-
0
0
-
-
-
0
1
-
-
0
0
0
0
0
1
-
-
0
0
0
-
-
u
-
-
0
0
0
0
0
;shl bl,1
注意:
1. 左移 N 次相当于乘于 2 的 N 次方,右左移 N 次相当于除乘于 2 的 N 次方。
2. 移位指令根据是否移入“1”到 CF,设置 CF,根据移位后的结果影响 SF,ZF,PF。根据最高符
号位是否改变设置 OF,如改变 OF=1.
3. ‘ u ’表示无定义,‘ - ’表示无影响。
2.13; (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,bx
(2)
数字 0~9 的 ASCII 码是:30h~39h
非压缩 BCD 码的 0~9 是:00h~09h
方法一:
and al,0fh
or al,30h
方法二:
xor al,30h
;即高 4 位为 3,则变为 0;高 4 位为 0,则变为 3
mov cl,4
again: shr dx,1
;采用“sar dx,1”,则实现算术右移
;实现 ASCII 到非压缩 BCD 码的转换
;实现非压缩 BCD 码到 ASCII 的转换
;求反 D5D4 位,其他不变
;实现逻辑右移
;shl bx,1
rcr ax,1
dec cl
jnz again
2.14; (1)用 sar 编写
2.20; 8086 的条件转移的转移范围:在当前指令地址的 +127---- -128 之内。
如条件转移的转移范围超出此范围,可在此范围内安排一条无条件转移,再转移到范围外的
目标地址。
2.21; (1)JMP Bx
(2)JMP tABLE[Bx] ;转移的有效地址 EA=[ds:20a1h+1256h]=[232f7]=3280h
(3)JMP [Bx][si] ;转移的有效地址 EA=[ds:1256h+528fh]=264e5h=2450h
2.22; (1) xor ax,1e1eh
;转移的有效地址 EA=BX=1256h
je equal
;AX=1e1eh(异或后为 0)
(2)test al,10000001b
jnz there
;AL 的 D0 或 D7 至少有一位为 1
(3) cmp cx,64h
jb there
;CX(无符号数)< 64h
2.23;
不循环,因为一进入循环就判 cx=0? 如 cx=0 就退出循环
mov cx,0
delay:loop delay
2.24; (1)若 DX > DI,转到 above 执行
cmp dx,di
ja above
;=jnbe above
(2)若 AX > SI,转到 greater 执行
cmp ax,si
jg greater
;=jnle greater
(3)若 CX = 0,转到 zero 执行
cmp cx,0
jz zero
jcxz zero
(4)若 AX-SI 产生溢出,转到 overflow 执行;
cmp ax,di
jo overflow
(5)若 SI≤AX,转到 less_eq 执行;
cmp si,ax
cmp ax,si
jle less_eq
jge less_eq
(6)若 DI≤DX,转到 below_eq 执行。
cmp di,dx
cmp dx,di
jbe below_eq
jae below_eq
2.25; 答:将首地址为 array 得 20 个字的数组求和,并将结果存入 total 单元中。
2.26; (1)
mov si,0
mov dl,string[si] ;第 1 个字符送 dl 寄存器
mov si,5
mov dh,string[si] ;第 6 个字符送 dh 寄存器
;si 清零
;BCD 码移到高半字节
;组合成压缩 BCD 码
;存入 dl 寄..
xor si,si
mov al,buffer[si] ;第一字节
inc si
mov ah,buffer[si] ;第二字节
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 寄..
(2)
(3)
(4)
(5)
test dx,0f000h
jz zero
mov ax,-1
jmp done
zero:
done:
mov ax,0
ret
lea bx,buffer1
lea dx,buffer2
mov cx,8
xor si,si
clc
;8 个字节
;si=0
;CF=0
;段地址
mov ax,0b800h
mov ds,ax
xor si,si
xor dx,dx
mov cx,99
mov ax,[si] ;第一个数
;地址偏移量 si=0
;和的高字 dx=0
;加的次数
again:
inc si
inc si
;指向下一个字单元
add ax,[si]
jnc
noc
inc dx
noc:
dec cx
jnz cx,again
ret
(6)
;加下一个数
;无进位转
;有进位 dx=dx+1
;次数-1
;非 0 继续加
mov si,offset string
mov cx,8000h
;32k=2^15=8000h
again: cmp [si],’$’
jnz next
mov [si],20h
next:
inc si
loop again
(7)
xor si,si
mov cx,100
again: dec array[si]
dec cx
jnz
again
(8)
;if [si]=’$’ [si]<-- ’ ’
;si<--0
;循环次数
;si<--0
cmp string[si],’$’
done
xor si,si
coun:
je
inc si
jmp coun
done: ret
and ax,ax ;or ax,ax
xor ax,ax ;
and ax,0 ;mov ax,0
clc ;
2..27; (1)使 CF=0 :
(2)使 AX=0 :
(3)同时使 AX=0 和 CF=0: and ax,0 ;xor ax,ax ;sub ax,ax
2.29; 压缩 BCD 码加法:AX←AX+BX
出口参数:AX=BCD 码和
2.24; okmsg
db ‘OK’, ‘$’errmsg
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
mov dx,errmsg
mov ah,9
int 21h
db ‘OK’, ‘$’
;错误 1:数据定义在代码中
overflow:
errmsg db ‘Error ! Overflow !’, ‘$’
mov dx,errmsg
mov ah,9
int 21h
; 错误 2:缺少 JMP 指令
; 键盘输入字符串
2.37; ;xt237.asm
.model small
.stack
.data
array db 255
db 0
array1 db 255 dup('$')
array2 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: inc bx
jmp again
done: mov dx,offset array1
mov ah,09h
int 21h
.exit 0
end
; 小于 a 和大于 z 的字符不是小写字母
; 在 a 和 z 之间的字符才是小写字母,转换为大写
; 保存到原位置
第三章(01)
3.1; 硬指令:每个硬指令就是一个处理器指令,在 CPU 执行时产生相应功能;
伪指令:伪指令并不产生处理器指令,它通常用于辅助汇编程序对源程序进行汇编。
3.2;
3.5; ⒈ 编辑
⒉ 汇编
⒊ 连接
⒋ 调试
3.6; ;xt236.asm 简化段定义格式
目标模块文件.obj
可执行文件.exe 或.com
应用程序
汇编程序
连接程序
调试程序
汇编语言源程序.asm
文本编辑程序
.model small ;定义程序的存储模式(小模式)
;定义堆栈段(默认 1024 个字节)
.stack
.data
;定义数据段
str1
str2
ab ’Input Number:0~9 : ’,0dh,0ah,’$’
ab ’Error!’,0dh,0ah,’$’
;定义代码段
;说明程序的起始点,建立 ds,ss 的内容。
;显示 str1 字符串
.cade
.startup
mov ah,09h
mov dx,offset str1
int 21h
gtekey: mov ah,1
;调用 BIOS 判断按键功能
al,‘0’
getkey
int 16h
jz
cmp
jb error
cmp al, ‘9 ’
ja error
mov ah,02h
mov dl,al
int 21h
.exit 0
error: mov ah,09h
;如 zf = 0,无键按下,等待
;有键按下,键值与‘0’比较
;如 < ‘0’,出错处理
;有键按下,键值与 ‘9’比较
;如 〉‘9’,出错处理
;调用 DOS 显示字符功能,显示该数字
;终止程序执行,返回 DOS
; 出错,调用 DOS 功能显示 str2 字符串
mov dx,offset str2
int 21h
jmp getkey
end
3.7; ;xt307.asm
stack
segment
;等待按键
; 汇编结束
dw 512 dup(?)
ends
segment
stack
data
array db 255
db 0
array1 db 255 dup('$')
array2 db 0dh,0ah,'$'