汇编语言程序设计(钱晓捷)课后答案

合集下载
  1. 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
  2. 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
  3. 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。

汇编语言程序设计(第二版) 钱晓捷习题答案
第二章(01)
2.1
(1)AX=1200h
(2)AX=0100h
(3)AX=4C2Ah
(4)AX=3412h
(5)AX=4C2Ah
(6)AX=7856h
(7)AX=65B7h
2.2(1) 两操作数类型不匹配
(2) IP指令指针禁止用户访问
(3) 立即数不允许传给段寄存器
(4) 段寄存器之间不允许传送
(5) 两操作数类型不匹配
(6) 目的操作数应为[ BP ]
(7) 源操作数应为[BX+DI]
(8) 立即数不能作目的操作数
2.3
lea bx,table ;获取table的首地址,BX=200H
mov al,8 ;传送欲转换的数字,AL=8
xlat ;转换为格雷码,AL=12H
2.4
堆栈是一种按“先进后出”原则存取数据的存储区域。

堆栈的两种基本操作是压栈和出栈,对应的指令是PUSH和POP。

2.5
mov ax,8057h
push ax
mov ax,0f79h
push ax
pop bx ;bx=0f79h
pop [bx] ;DS:[0f79h]=8057h
2.6
AL=89h CF ZF SF OF PF
AL=12h 1 0 0 1 1
AL=0afh 0 0 1 0 1
AL=0afh 1 0 1 0 1
AL=00h 0 1 0 0 1
AL=0ffh 0 0 1 0 1
AL=00h 0 1 0 0 1
2.7 W=X+Y+24-Z
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
mov ax,X ;ax=A
imul Y ;dx,ax = A*B (将操作数看作符号数,以下同)
mov cx,ax
mov bx,dx ;bx,ax <-- dx,ax =A*B
mov ax,Z ;ax = C
cwd ;dx,ax =C (扩展符号后为双字)
add cx,ax
adc bx,dx ;bx,cx <-- bx,cx+dx,ax=A*B+C
sub cx,540
sbb bx,0 ;bx,cx<-- A*B+C-540
mov ax, V ;ax= D
cwd ;dx,ax= D (扩展符号后为双字)
sub ax, cx
sbb dx, bx ;dx,ax = dx,ax - bx,cx = D-(A*B+C-540)
idiv X ;运算结果:[D-(A*B+C-540h)]/A ;ax存商,dx存余数
2.10;(1)xchg的操作数不能是立即数
(2不能对CS直接赋值
(3)两个操作数不能都是存储单元
(4)堆栈的操作数不能是字节量
(5)adc的操作数不能是段寄存器
(6)没有确定是字节还是字操作
(7)in不支持超过FFH的直接寻址
(8)out只能以AL/AX为源操作数
第二章(02)
2.11;
指令AX的值CF OF SF ZF PF
Mov ax,1407h1470h-----
And ax,ax1470h00000
Or ax,ax1470h00000
Xor ax,ax000011
Not ax0ffffh-----
Test ax,0f0f0h0ffffh00101
注意: 1. mov, not指令不影响标志位
2. 其他逻辑指令使CF=OF=0, 根据结果影响其他标志位。

第2章(03)
2.12;
指令注释执行结果CF OF SF ZF PF
mov si,ax si=ax si=0008h - - - - -
shl si,1 si=2*ax si=0010h 0 0 0 0 0
add si,ax si=3*ax si=0018h 0 0 0 0 1
mov dx,bx dx=bx dx=0010h - - - - -
mov cl,03h cl=03h - - - - -
shl dx,cl dx=8*bx dx=0080h 0 u 0 0 0
sub dx,bx dx=7*bx dx=0070h 0 0 0 0 0
add dx,si dx=7*bx+3*ax dx=0088h 0 0 0 0 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 ;shl bl,1
add al,bl
;考虑进位xor ah,ah
mov bx,ax
mov cl,3
shl ax,cl
add ax,bx ;shl bx,1
add ax,bx
(2)
数字0~9的ASCII码是:30h~39h
非压缩BCD码的0~9是:00h~09h
方法一:
and al,0fh ;实现ASCII到非压缩BCD码的转换
or al,30h ;实现非压缩BCD码到ASCII的转换
方法二:
xor al,30h ;求反D5D4位,其他不变
;即高4位为3,则变为0;高4位为0,则变为3
mov cl,4
again: shr dx,1 ;实现逻辑右移
;采用“sar dx,1”,则实现算术右移
rcr ax,1
dec cl
jnz again
2.14; (1)用sar编写
2.20; 8086的条件转移的转移范围:在当前指令地址的+127---- -128之内。

如条件转移的转移范围超出此范围,可在此范围内安排一条无条件转移,再转移到范围外的目标地址。

2.21; (1)JMP Bx ;转移的有效地址EA=BX=1256h
(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
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;
mov cx,0 不循环,因为一进入循环就判cx=0? 如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 jcxz zero
jz 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寄存器
(2)
xor si,si ;si清零
mov al,buffer[si] ;第一字节
inc si
mov ah,buffer[si] ;第二字节
mov cl,4
shl ah,cl ;BCD码移到高半字节
or al,ah ;组合成压缩BCD码
mov dl,al ;存入dl寄..
inc si
mov al,buffer[si] ;第三字节
inc si
mov ah,buffer[si] ;第四字节
mov cl,4
shl ah,cl ;BCD码移到高半字节
or al,ah ;组合成压缩BCD码
mov dh,al ;存入dh寄..
(3)
test dx,0f000h
jz zero
mov ax,-1
jmp done
zero: mov ax,0
done: ret
(4)
lea bx,buffer1
lea dx,buffer2
mov cx,8 ;8个字节
xor si,si ;si=0
clc ;CF=0
(5)
mov ax,0b800h
mov ds,ax ;段地址
xor si,si ;地址偏移量si=0
xor dx,dx ;和的高字dx=0
mov cx,99 ;加的次数
mov ax,[si] ;第一个数
again: inc si ;指向下一个字单元
inc si
add ax,[si] ;加下一个数
jnc noc ;无进位转
inc dx ;有进位dx=dx+1
noc: dec cx ;次数-1
jnz cx,again ;非0继续加
ret
(6)
mov si,offset string
mov cx,8000h ;32k=2^15=8000h
again: cmp [si],’$’
jnz next
mov [si],20h ;if [si]=’$’ [si]<-- ’ ’
next: inc si
loop again
(7)
xor si,si ;si<--0
mov cx,100 ;循环次数
again: dec array[si]
dec cx
jnz again
(8)
xor si,si ;si<--0
coun: cmp string[si],’$’
je done
inc si
jmp coun
done: ret
2..27; (1)使CF=0 :clc ;and ax,ax ;or ax,ax
(2)使AX=0 :xor ax,ax ;and ax,0 ;mov 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.34; 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 db ‘OK’, ‘$’
mov dx,errmsg ;错误1:数据定义在代码中
mov ah,9
int 21h
overflow: errmsg db ‘Error ! Overflow !’, ‘$’
mov dx,errmsg ; 错误2:缺少JMP指令
mov ah,9
int 21h
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' ; 小于a和大于z的字符不是小写字母
jb next
cmp al,'z'
ja next
sub al,20h ; 在a和z之间的字符才是小写字母,转换为大写mov [bx],al ; 保存到原位置
next: inc bx
jmp again
done: mov dx,offset array1
mov ah,09h
int 21h
.exit 0
end
第三章(01)
3.1; 硬指令:每个硬指令就是一个处理器指令,在CPU执行时产生相应功能;伪指令:伪指令并不产生处理器指令,它通常用于辅助汇编程序对源程序进行汇编。

3.2;
3.5; ⒈编辑文本编辑程序汇编语言源程序.asm
⒉汇编汇编程序目标模块文件.obj
⒊连接连接程序可执行文件.exe或.com
⒋调试调试程序应用程序
3.6; ;xt236.asm 简化段定义格式
.model small ;定义程序的存储模式(小模式)
.stack ;定义堆栈段(默认1024个字节)
.data ;定义数据段
str1 ab ’Input Number:0~9 : ’,0dh,0ah,’$’
str2 ab ’Error!’,0dh,0ah,’$’
.cade ;定义代码段
.startup ;说明程序的起始点,建立ds,ss的内容。

mov ah,09h ;显示str1字符串
mov dx,offset str1
int 21h
gtekey: mov ah,1 ;调用BIOS判断按键功能
int 16h
jz getkey ;如zf = 0,无键按下,等待
cmp al,‘0’;有键按下,键值与‘0’比较
jb error ;如< ‘0’,出错处理
cmp al, ‘9 ’;有键按下,键值与‘9’比较
ja error ;如〉‘9’,出错处理
mov ah,02h ;调用DOS显示字符功能,显示该数字
mov dl,al
int 21h
.exit 0 ;终止程序执行,返回DOS
error: mov ah,09h ;出错,调用DOS 功能显示str2字符串
mov dx,offset str2
int 21h
jmp getkey ;等待按键
end ;汇编结束
3.7; ;xt307.asm
stack segment
dw 512 dup(?)
stack ends
data segment
array db 255
db 0
array1 db 255 dup('$')
array2 db 0dh,0ah,'$'
data ends
code segment 'code'
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' ; 小于a和大于z的字符不是小写字母
jb next
cmp al,'z'
ja next
sub al,20h ; 在a和z之间的字符才是小写字母,转换为大写
mov [bx],al ; 保存到原位置
next: inc bx
jmp again
done: mov dx,offset array1
mov ah,09h
int 21h
mov ax,4c00h
int 21h
code ends
end start
3.9; (1) mov byte ptr [bx],1000 ;1000超出了一个字节范围
(2) mov bx,offset myword[si];寄存器的值只有程序执行时才能确定,;而offset是汇编过程计算偏移地址,故无法确定
;可以改为lea bx,myword[si]
(3) cmp mybyte1,mybyte2 ;两个都是存储单元,指令不允许
(4) mov al,mybyte1+mybyte2 ;变量值只有执行时才确定,汇编过程不能计

(5) sub al,myword ;字节量AL与字量myword,类型不匹配
(6) jnz myword ;Jcc指令只有相对寻址方式,不支持间接寻址方式
3.10; 前者为“与”操作硬指令助记符,可汇编成机器代码。

后者为逻辑运算符,在汇编时进行“与”运算,产生具体数值。

3.11; 注:对于逻辑运算,有关操作数可化为二进制数。

(1)mov al,23h AND 45h OR 67h ; 67h
(2)mov ax,1234h/16 + 1Oh ; 133h
(3)mov ax,NOT(65535 XOR 1234h) ; 1234h
(4)mov al, LOW 1234h OR HIGH 5678h ; 76h
(5)mov ax,23h SHL 4 ; 0234h
(6)mov ax, 1234h SHR 6 ; 0048h
(7)mov al,’a’ AND (NOT(’a’-’A’)) ; 41h
(8)mov al,’H’ OR 00100000b; 68h
(9)mov ax,(76543 LT 32768) XOR 7654h ; 7654h
3.12; 假设block开始的数据块有32个字节数据:16个正数+100 (64h),16个负数 -48 (0doh)
分别连续分布:
block db 16 dup(100),16 dup(-48);也可以是任意字节数据,随意分布。

dplus db 32 dup(?) ;为正数预留存储空间
dminus db 32 dup(?) ;为负数预留存储空间
count equ
32 ;字节数
3.15; mydataseg segment
my1b db ‘Personal Computer’
my2b db 20
my3b db 14h ;20h
my4b db 00010100b
my5w dw 20 dup(?)
my6c equ 100 ;my6c = 100
my7c equ <Personal Computer>
mydataseg ends
3.18; 段地址:表示标号所在代码段的段地址;
偏移地址:表示标号所在代码段的段内偏移地址;
类型:引用该标号时,表示它所在同一个段――near类型,还是另外一个
段――far类型。

3.19; mydata segment
ORG lO0h
VARW DW l234H,5678H
VARB DB 3,4
AGLIN 4
VARD DD 12345678H
EVEN
BUFF DB 10 DUP(?)
MESS DB ’HELLO’
MOV AX, OFFSET VARB + OFFSET MESS ;AX = 4+16H = 1AH
MOV AX, TYFE TYPE BUFF+TYPE MESS+TYPE VARD;AX = 1+1+4 = 06H
MOV AX,SIZE VARW+SIZE BUFF+SIZE MESS;AX = 4+10+5 = 19 = 13H
MOV AX,LENGTH VARW + LENGTH VARD ;AX = 2+1 = 03H
MOV AX,LENGTH BUFF + SIZE VARW ;AX = 10+4 =14 = 0EH MOV AX,TYPE BIGIN ;AX = FF02H (近)
MOV AX,OFFSET BEGIN ;AX =
1BH
3.22; 段定义伪指令段名定位组合类
别组名
.CODE _TEXT WORD PUBLIC ’CODE

.DATA _DATA WORD PUBLIC ’DATA
’ DGROUP
.STACK STACK PARA STACK ’STACK’ DGROUP
3.25; .model small
.stack
.data
num equ 5
datalist dw -1,0,2,5,4,?
.code
.startup
mov bx,offset datalist
mov cx,num
xor ax,ax
again: add ax,[bx]
inc bx
inc bx
loop again
mov [bx],ax
.exit 0
end
3.26; stack segment para ‘stack’
dw 512 dup(?)
stack ends
data segment
dword array db 100 dup(?)
data ends
code segment ‘code’
assume cs:code,ds:data,es:data,ss:stack
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
end start
3.27;
解答
;xt327.asm
.model small
.stack 256 ;定义堆栈段大小为256个字节
.data
anum dd 11223344h ;定义两个双字的数(随意)
bnum dd 77553311h
sum dd ? ;定义结果,执行后为:88776655h .code
.startup
xor si, si ;相对于变量的位移量清零
mov cx, 2 ;分高低字分别相加,共两次
clc ;清零cf
again:mov ax, anum[si] ;取第一个数的一个字(先低字
后高字)
adc ax, bnum[si] ;取第二个数的一个字(先低字后
高字)
mov sum[si], ax ;存和的一个字(先低字后高字)
inc si ;修改位移量指向下一个字(加2)
inc si
loop again ;cx=cx-1 ,if cx<>0 ,jump again
.exit 0
end
3.28; .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
end
4.3; 思路:设这四组从低位到高位分别放在AL、BL、CL和DL寄存器中。

这里仅列出代码段:
mov bl, al ;将al中的两组分开
and al, 0fh ;屏蔽高四位后送al
mov cl, 4 ;原al中的数据逻辑右移4次送bl
shr bl, cl
mov dl, ah ;将ah中的两组分开
and dl, 0f0h ;屏蔽低高四位后送dl
mov cl, 4 ;原ah中的数据逻辑右移4次送dl
shr dl, cl
mov cl, ah ;屏蔽高四位后送cl
and cl, 0fh
4.4; getkey: mov ah, 1 ;从键盘输入,出口:al存键值
int 21h
cmp al, ’a’;判键值是小写字母?
jb getkay
cmp al, ’z’
ja getkay
sub al,20h ;是小写字母转换为大写字母
mov ah, 09h ;显示
int 21h
4.8; (1)将jmp table[bx]指令改为:mov dx, table[bx]
(2)去掉源程序中:.exit 0---end之间的语句
4.17; .model small
.stack 256
.data
stri1 db ’please input number:1--9’,odh,oah,’$’
.code
.startup
again: mov dx,offset stri1 ;显示stri1,提示输入mov ah,09h
int 21h
mov ah,01h ;调用输入一个字符
int 21h ;输入一个字符存在al中
cmp al, ’1’;判该字符,如不在‘1’--‘9’
jb again ;重新输入
cmp al, ’9’
ja again
and al,0fh ;在‘1’--‘9’,屏蔽高4位
mov cx, al ;振铃次数送cx
.repeat
mov dl, 07h ;调用一次振铃
mov ah, 02h
int 21h
mov dx ,0ffffh ;延时
abc: dec dx
jnz abc
.untilcxz ;cx=cx-1,cx=0 退出
.exit 0
end
4.22;crazy PROC ;crazy PROC
push ax ;
xor ax,ax ;xor ax,ax
xor dx,dx ;xor dx,dx
again: add ax,[bx] ;again: add ax,[bx]
adc dx,0 ;adc dx,0
inc bx ;inc bx
inc bx ;inc bx
loop again ;loop again
ret ;ret
ENDP crazy ;crazy ENDP
4.25;子程序中又调用子程序就形成子程序嵌套。

子程序中直接或间接调用该子程序本身就形成子程序递归。

4.26;.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 bx
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;astob proc
and dh, 0fh ;十位数的ASCII码转为二进制数
mov al, dh
mul 10 ;al= 10*dh
and dl, 0fh ;个位数的ASCII码转为二进制数
add al, dl ;al= 10*dh + dl
ret
astob endp
5.1;cmp X, 5
je abc
jmp done
abc: cmp ax, bx
jne cde
jmp done
cde: inc ax
done:....
5.2;cmp X, 5
je abc
jmp done
cmp ax, bx
jne abc
jmp done
abc: inc ax
done: ....
5.4;
宏定义采用一对伪指令实现,格式如下(其中方括号表示可选):
宏名macro [形参表]
宏定义体
endm
宏调用时,利用宏名带上实参即可,格式如下:
宏名[实参表]
5.6;宏的参数功能非常强大,即可以没有参数,也可以带有一个或多个参数;而且参数的形式非常灵活,可以是常量、变量、存储单元、指令或它们的一部分,还可以是表达式等。

宏展开时的形参与实参结合是用位置匹配的实参对形参进行取代。

实参个数与形参个数不同时,多余的实参没有作用,缺少的实参则用空格代替。

取代时,不进行有效性等检查。

5.7;宏汇编的特点是在汇编阶段进行语句展开,无需返回,不减少目标程序代码,执行速度没有改变;通过形参、实参结合传递参数。

宏汇编适合于程序段较短或要求执行速度快的场合。

子程序在执行阶段需要由主程序执行调用指令CALL进行调用,进而转入子程序执行,子程序执行结束执行返回指令RET返回主程序。

子程序如果被多次使用,则会减少目标程序代码;主程序与子程序之间主要采用寄存器、存储单元、堆栈等传递参数。

子程序适合于程序段较长或为减少目标代码的场合。

5.9;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 macro lcode,dopd,sopd
lcode dopd,sopd
endm
例如,如果使用“and ax,[bx]”指令,可以利用该宏定义,写出宏指令如下:logical and,ax,[bx]
初看起来,这个问题似乎比较难解决。

实际上,这4条逻辑指令都具有相同的指令操作数格式,所以可以归纳为如上宏定义。

5.12; move MACRO doprnd,soprnd
push ax
mov ax,soprnd
mov doprnd,ax
pop ax
ENDM
第五章(02)
5.13;
解答
movstr strN,dstr,sstr
lea si, sstr
lea di, dstr
mov cx, strN
cld
rep movsb
endm
5.14; ①ABSDIF Pl,P2,DISTANCE ②ABSDIF [BX],[SI],[DI]
1 push ax 答:1 push ax
1 mov ax,P1 1 mov ax,[bx]
1 sub ax,P
2 1 sub ax,[si]
1 cmp ax,0 1 cmp ax,0
1 jge ?? 0000 1 jge ?? 0000
1 neg ax 1 neg ax
1 ?? 0000 1 ?? 0000
1 mov DISTANCE, ax 1 mov [di],ax
1 pop ax 1 pop ax
5.15; wtemp =2
repeat 100
dw 0
dw wtemp
wtemp =wtemp+2
endm
5.18; DOS21H MACRO callnum,calladdress
mov ah,callnum
ifnb
mov dx,calladdress
endif int 21h
ENDM
5.22; 1.编写主程序,并进行汇编产生OBJ模块;
2.编写独立的子程序,也进行汇编产生OBJ模块;
3.分别编写主程序与子程序时,需要处理好共用变量、过程、逻辑段属性、参数传递等问题;
4.利用连接程序将主程序模块和子程序模块连接起来,形成完整的可执行程序。

5.28; ;数据段
table dw 2277,2138,1808,1709,1522,1356,1208,1139
;对应中音1 ~ 7和高音i的定时器记数值
;代码段
mov al,0b6h ;设置定时器2工作方式
out 43h,al
again: mov ah,1 ;等待按键
int 21h
cmp al,’1’;判断是否为数字1~8
jb next
cmp al,’8’
ja next
sub al,30h
;1~8的ASCII码转换为二进制数
sub al,1
;再减1,将数字1~8变为0~7,以便查表
xor ah,ah
shl ax,1 ;乘以2
mov bx,ax
;记数值表是16位数据,无法采用xlat指令
mov ax,table[bx] ;取出对应的记数值
out 42h,al ;设置定时器2的记数值
mov al,ah
out 42h,al
in al,61h ;打开扬声器声音
or al,03h ;使D1D0=PB1PB0=11b,其他位不变
out 61h,al
jmp again ;连续发声,直到按下另一个键
next: push ax
in al,61h ;不是数字1~8,则关闭扬声器声音
and al,0fch ;使D1D0=PB1PB0=00b,其他位不变
out 61h,al
pop ax
cmp al,1bh ;判断是否为ESC键(对应ASCII码1bh)
jne again ;不是ESC,继续;否则程序执行结束
------------------------------------------------------------------------------------------------------- 1。

.(1)立即寻

没有
(2)直接寻

7237H
(3)使用BX的寄存器寻

没有
(4)使用BX的间接寻

637DH
(5)使用BX的寄存器相对寻

0D5F0H
(6) 基址变址寻

8E18H
(7)相对基址变

004FH
2。

根据下列要求,写出相应的汇编指令。

(1)ADD DX,BX
(2) ADD AL,[BX][SI]
(3) ADD [BX][0B2H], CX
(4) MOV AX,2A59H ADD [0524H] ,AX
(5) ADD AL ,0B5H
3。

(1)寄存器间接寻址
MOV BX,OFFSET BLOCK
ADD BX,OAH
MOV DX ,[BX]
(2)寄存器相对寻址
MOV BX,0AH
MOV DX,BLOCK[BX]
(3)基址变址寻址
MOV BX , OFFSET BLOCK
MOV SI,OAH
MOV DX,[BX][SI]
4。

现有(DS)
=2000H,(BX)=0100H,(SI)=0002H,(20100)=12H,(20101)=34H,(20102)=56H,(201 03)=78H,(21200)=2AH,(20201)=4CH,(21202)=B7H,(21203)=65H,试说明下列各条指令执行完后,AX寄存器的内容。

(1)
MOV AX,1200H
1200H
(2) MOV AX,BX
0100H
(3) MOV AX,[1200]
4C2AH
(4)
MOV AX,[BX]
3412H
(5)
MOV 1100[BX]
4C2AH
(6) MOV AX,[BX][SI]
7856H
(7) MOV AX,1100[BX][SI]
65B7H
5。

(1)7CD9H (2) (212A0+1200)
H=0600H (3) (212A0H+1200H+5119H)H=098AH
6。

MOV BX,2000H 或LES BX,[2000]
LES DI ,[BX]
MOV AX,ES:[BX]
MOV AX, ES:DI
7。

转向地址OBJ的值分别为:(1)
064DH (2)0691H (3)05ECH 注意有符号数的符号位~~
8。

(1)MOV AX,0ABH
立即数寻

无物理地址
(2)
MOV AX,BX
寄存器寻

同上
(3)
MOV AX,[100]
直接寻

20100H
(4)
MOV AX,VAL
直接寻

20050H
(5) MOV AX,[BX]
寄存器间接寻

20100H
(6) MOV AX,ES:
[BX] 直接寻

21100H
(7) MOV AX,[BP]
寄存器间接寻

20010H
(8)
MOV AX,[SI]


200A0H
(9) MOV AX,[BX+10]
寄存器相对寻

20110H
(10)
MOV AX,VAL[BX] 同

20150H
(11) MOV AX,[BX][SI]
基址变址寻

201A0H
(12) MOV AX,VAL[BX][SI]
相对基相变址寻
址201 F0H
9。

(1)的指
令:MOV AX, [BX][0CH]
MOV ZREO ,AX
(2) 的指
令:MOV AX,ARRAY[BX]
MOV ZREO,AX
10。

MOV AX,TABLE 是把符号地址TABLE里的内容送到AX 里,指令执行完后,(AX)=1234H
LEA AX,TABLE 是把符号地址TABLE 的有效地址(这里是偏移量)送到指定寄存器AX里,指
令执行完后,(AX)=0032H
11。

执行完指令后,(AX)=1E00H
12。

LEA AX,CSTRING
MOV DL,AX
MOV DH,[AX+6]
13。

这个嘛不想做了,麻烦啊~
14。

LES BX,[2000]
MOV AX,ES:[BX]
注意:一开始我写的是:MOV AX,ES:[8000H],后来看到题目里要求写两条指令,所以一条就不行了,就要画蛇添足~~~
15。

运算结

SF,ZF,CF,OF各位的值(从左至右依次往下~)
(1)74D4H
0 0 0 0
(2) A5C1H
1 0 0 1
(3) 4240 H
0 0 1 0
(4) 0000 H
0 1 1 0
16。

接上表~
(1) 5D14
0 0 0 0
(2) A870
1 0 0 1
(3) D870
1 0 0 0
(4) 34E4
0 0 1 0
17。

(1).MOV AX,Z ;把Z送到AX
SUB AX,X ;Z-X
ADD AX,W ;W+(Z-X)
MOV Z,AX ;结果保存在Z中(2). MOV AX,R ;R送到AX中
ADD AX,9 ;R+9
MOV BX,X ;X送到BX中
ADD BX,6 ;X+6
ADD AX,BX ;(R+9)+(X+6),结果保存在AX中
MOV BX,W ;W送到BX中
SUB BX,AX ;W-(R+9)-(X+6),连续减去两个数,相当于减去这两个数的和~~
MOV Z,BX ;最终结果送到Z中
(3)
MOV AX,W ;把W送到AX
IMUL X
;W*X 注意:低十六位保存在AX里,高十六位保存在DX里
MOV BX,Y ;把Y送到
BX 里
ADD BX,6 ;Y+6
IDIV BX ;(W*X)/(Y+6) 注意:商保存在AX里,余数保存在DX里
MOV Z,AX ;把商送到Z 中,
MOV R,DX ;把余数送到R中
(4)
MOV AX,W ;把W送到AX中
SUB AX,X ;W-X
CWD ;把AX
中的十六位带符号数扩展成为32位,低十六位保存在AX中,高十六位保;在DX中
MOV BX,5 ;把5送到BX中
IDIV BX ;(W-X)/5 被除数为32位,除数为16位商为16位
IMUL Y ;上一步结果再乖以Y,积为32位
MOV BX,2 ;把2送到BX 中
IMUL BX ;上一步乘法所得的积再乘以2,各为64位
MOV Z,EAX ;把结果中低32位放入Z双字单元中
MOV [Z+4],EDX ;把结果中高32位放入Z+4双字单元中
18。



本条指令执行完后AX的内

执行完后CF,SF,ZF的值
MOV AX,1234H 1234H
0 0
MOV CL,4
1234H
0 0 0
ROL AX,CL 2 341H
1 0 0
DEC AX
2340H
1 0 0
MOV CX,4
2340H
1 0 0
MUL CX
8918H
0 0 0
INT 20H
程序结束后,(AX)=8918H (DX)=0000H
19。



执行完后AX的内
容执行完后CF ,SF,ZF和OF的值
MOV AX,0
001 0
DEC AX
FFFFH
0100
ADD AX,7FFFH
7FFEH
0000
ADD AX,2
8000H
0101
NOT AX
7FFFH
0101
SUB AX,0FFFFH
8000H
1101
ADD AX,8000H
0000H
1011
SUB AX,1
FFFFH
1101
AND AX,58D1H
58D1H
0000
SAL AX,1
B1A2H
0101
SAR AX,1
D8D1H
0100
NEG AX
272FH
1000
ROR AX,1
9397H
1001 20。

(1)MOV AX,DATAX
ADD DATAY,AX
(2)MOV AX,DATAX
ADD DATAY,AX
MOV AX,[DATAX+2]
ADD [DATAY+2] ,AX
(3)本組指令的作用是:BX <------[DATAX]+[DATAY]+[1](CF的
值) 帶進位加法~
(4)MOV AX,DATAX
MUL DATAY ;結果的高十六位保存在DX裏,低十六位保存在AX裏
(5)MOV AX,DATAX
MOV DX,[DATAX+2]
MUL DWORD DATAY ;結果的高三十二位保存在EDX裏,低三十二位保存在EAX裏
;這裏所用的DWORD為一屬性操作符,在第四章會學到~~
(6)MOV AX,DATAX
DIV 17H
;結果商保存在AL中,余數在AH中
(7)MOV AX,DATAX
CDW
DIV DATAY
21。

NEG DX
NEG AX
SBB DX,0
22。

MOV AX,A ;把雙字長數的低字放到AX中
MOV DX,[A+2] ;把雙字長數的高字放到DX中
TEST DX,8000H ;測試雙字長數的符號
JZ RIGHT
;如果是非負數,則直接保存
NEG DX
;如果是負數,則求補
NEG AX
SBB DX,
RIGHT:MOV B ,AX
MOV [B+2],DX
1。

指令下列指令的错误
(1)MOV AH,BX 源操作数与目的操作数长度不相同~
(2)MOV [BX],[SI] 这是两个内存单元在传递数据,而这在汇编语言中是不允许的~~~
(3) MOV AX,[SI][DI] 不能用两个变址连用,其中一个应为基址~
(4)MOV MYDAT[BX][SI],ES:AX 同(2)
(5)MOV BYTE PTR[BX],1000 必须有一个为寄存器
(6)MOV BX,OFFSET MYDAT[SI] OFFSET 不能与复杂操作数相连~
(7)MOV CS,AX 不能对CS赋值~
(8)MOV ECX,AX 操作数类型不同~
2。

(1)合法(2)非法(3)非法(4)非法
3。

(1)非法,操作数类型不同
(2)合法
(3)非法,不允许两个内存单元直接传递数据,你把人家CPU放哪了?~~~呵~
(4)合法~
(5)合法~
(6)合法
(7)非法~,必须有一个寄存器~~把CPU放哪了?~~
(8)非法,两个内存单元地址相加没有意义~
(9)合法
(10)非法,同(3)
4。

(1)两内存单元传递数据
(2)操作数类型不同
(3)跳转指令只能跳转到符号地址
(4)转移指令后不能为变量,只能为标号
(5)格式不对,应为NEAR PTR 。

5。

图,就不画了~~
6。

(1)DW 5150H
(2)MOV AX,5150H
MOV ADDRESS ,AX
等,写出几个就可以~~
7。

DATASG SEGMENT
FLD1B DB 'personal computer'
FLD2B DB 10 DUP (32)
FLD3B DB 20H
FLD4B DB 01011001B
FLD5B DB '32654'
FLD6B DB 10 DUP (0)
FLD7B DB 'PART1',DUP 20 (?),'PART2',DUP 50 (?),'PART3 ',DUP 14 (?)
FLD1W DW 0FFF0H
FLD2W DW 01011001B
FLD3W DW FLD7B
FLD4W DW 5,6,7,8,9
FLD5W DW 5 DUP (0)
FLD6W DW (FLD6W-FLD1W)-(FLD7B-FLD1B)
8。

PLENTH 的值为22,GE表示该数据段的长度~
9。

L的值为7
10。

(1)MOV AX,OFFSET LNAME
(2)MOV SI,WORD PTR CODE_LIST
(3)CODE_LENGHT EQU $-CODE_LIST
11。

代码如下:
DATA_SEG SEGMENT
DB 5
DATA_LIST DW 0FFFFH,0,2,5,4, 5 DUP (?)
DATA_SEG ENDS
CODE_SEG SEGMENT
ASSUME CS:CODE_SEG,DS:DATA_SEG
MOV AX,5
MOV MAX,AX
MOV AX,0FFFFH
MOV MIN,AX
.. CODE_SEG ENDS
12。

(1)10025
(2)19
(3)2548
(4)3
(5)103
(6)0FFFFH
(7)1
(8)3
13。

分别汇编成为:
MOV AX,0AH
MOV BL,0AH
MOV CL,1
14。

(1)(AX)=1 (2)(AX)=2 (3)(CX)=14H (4)(DX)=32H (5)(CX)=1]
15。

(1)操作符错误改为:DATA_SEG SEGMENT
(2)缺少段名在前面加个段名
(3)引用应该从;开始把那个“/”改成“;”
(4)结构混乱改为:
MAIN_PROC PROC FAR。

MAIN_PROC ENDP
END
16。

程序框架如下:
DATA_SEG SEGMENT
;定義數據段
ORG 0E000H
;讓下一字節地址為0E000H
FIRST_TYPE EQU THIS WORD ;建立一個與下一存儲單元有相同段地址和偏移地址的操作數
DATA_GROUP DB 100 DUP(?) ;給DATA_GROUP 分配100個字節的空間
DATA_SEG ENDS
;數據段結束
STACK_SEG SEGMENT
;定義堆棧段
ALIGN 16
;讓下一字節從小段首地址開始
STACK DB 100 DUP (?)
;為堆棧分配空間
STACK_SEG ENDS

CODE_SEG SEGMENT
ASSUME CS:CODE_SEG ,DS:DATA_SEG ,SS:STACK_SEG
ORG 1000H
;讓下條指令的第一個字節從1000H開始
START : MOV AX,DATA_SEG
MOV DS,AX
MOV AX,STACK_SEG
MOV SS,AX
.......
CODE_SEG ENDS
END START
;結束~ 17。

程序如下:
D_SEG SEGMENT
;数据段定义
AUGEND DD 99251
D_SEG ENDS
E_SEG SEGMENT
;附加段定义
ADDEND DD -15926
E_SEG ENDS
C_SEG SEGMENT
;代码段定义
ASSUME CS:C_SEG , DS:D_SEG,ES:E_SEG START: MOV AX,D_SEG
MOV DS,AX
MOV AX,E_SEG
MOV ES,AX
MOV EAX,DS:AUGEND
ADD EAX,ES:ADDEND
MOV DS:SUM,EAX
C_SEG ENDS
END START
;结束
18。

19。

需要加上PTR伪操作的有:(3)(4);
(2)不需要加,因为[BX]自动转化为字节类型
5.1 试编写一个汇编语言程序,要求对键盘输入的小写字母用大写字母显示出来
.model small
.stack 100H
.code
main proc far
Start:
push dx
mov ax,0
push ax
mov ah,01H
int 21H
cmp al,61H
jl lop1
cmp al,7BH
jge lop1
sub al,20H
mov dl,al
lop1:mov ah,2
int 21H
ret
End start
5.2 编写一个程序,从键盘接受一个小写字母,然后找出它的前导字符和后续字符,再按顺序显示这三个字符.
.model small
.386
.code
MAIN PROC FAR
start:
push ds
sub ax,ax
push ax
input:
mov ah,7H
int 21H
cmpe:
cmp al,20H
JE exit
cmp al,61H
JL input
cmp al,7AH
JG input
print:
sub al,01H
mov dl,al
mov ah,02H
int 21H
add al,01H
mov dl,al
mov ah,02H
int 21H
add al,01H
mov dl,al
mov ah,02H
int 21H
;输出换行回车
mov dl,0AH
mov ah,02H
int 21H
jmp input
exit :
ret
MAIN ENDP
END start
5.3 将AX寄存器中的16位数分成4组,每组4位,然后把这四组数分别放在AL、BL、CL、DL中。

.model small
.data
x DB 4 dup(?)
.stack 100H
.code
MAIN PROC FAR
start:
push ds
and ax,00H
push ax
mov ax,1234H
mov cx,04H
mov si,00H
lop:
push cx
mov cl,04H
rol ax,cl
push ax
and al,0FH
mov BYTE PTR x[si],al pop ax
pop cx
add si,2
Loop lop
mov al,[X]
mov bl,[X+2]
mov cl,[X+4]
mov dl,[X+6]
print:
mov ah,02H
int 21H
mov dl,al
mov ah,02H
int 21H
mov dl,bl
mov ah,02H
int 21H
mov dl,cl
mov ah,02H
int 21H
exit:
MAIN ENDP
END start
5.4 编写一个程序,要求比较两个字符串STRING1和STRING2所含字符是否相同,若相同则显示"MATCH",若不同在显示:"NO MATCH"。

.model small
.data
STRING1 DB 'I must study hard'
STRING2 DB 'I must study hard'
X DB 'MATCH .$'
Y DB 'NO MATCH.$'
.stack 100H
.code
main proc far
Start:
push dx
mov ax,0
push ax
mov ax,@data
mov ds,ax
mov es,ax
lea si,STRING1
lea di,STRING2
cld
mov cx,(string2-string1)/ type STRING2
ret
End start
5.5编写一个程序,要求能从键盘接收一个个位数N,然后响铃N次,(响铃的ASCII为07)。

这题比较有意思,呵呵。

model small
.code
main proc far
start:
push ds
sub ax,ax
push ax
mov ah,01H
int 21H
cmp al,20H
je exit
cmp al,30H
jle input
cmp al,3AH
jge input
sub al,30H
mov cx,ax
and cx,0FFH
lop:
mov dl,07H
mov ah,2H
int 21H
loop lop
jmp input
exit :
ret
main endp
end start
5.6编写程序,将一个包含有20个数据的数组M分成两个数组:正数数组和负数数组N,并分别把这两个数组中的数据个数显示出来。

.model small
.386
.data
M DB 1,2,3,4,5,6,7,8,9,10,-1,-2,-3,-4,-5,-6,-7,-8,-9,-10
P DB 20 dup(?)
N DB 20 dup(?)
.code
MAIN PROC FAR
start:
PUSH ds
AND ax,0
PUSH ax
MOV ax,@data
mov ds,ax
MOV bx,0
MOV si,0
MOV di,0
MOV cx,20D
cmpe :
CMP M[bx],0
JL ifLow
hight :
MOV al,byte ptr M[bx]
MOV byte ptr P[si],al
inc si
JMP commonContrl
ifLow :
MOV al,byte ptr M[bx]
MOV byte ptr N[di],al
INC di
commonContrl:
INC bx
loop cmpe
mov ax,si
call showNumber
call printCR
mov ax,di
call showNumber
exit: ret
MAIN ENDP
;;;;;;;;;;;;;;;;;注意,ShowNumber这个子过程有问题,请自行修改;;;;;;;;;;;;;;;;
showNumber PROC NEAR
covernum:
DAA
MOV dx,ax
MOV cl,2H
show:
ROL dl,4H
push dx
AND dl,0fH
ADD dl,30H
MOV ah,02H
int 21H
pop dx
loop show
ret
showNumber ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
printCR PROC NEAR
mov dl,0aH
mov ah,02H
int 21H
mov dl,0dH
mov ah,02H
int 21H
ret
printCR ENDP
end start
5.7编制一个汇编语言程序,求出首地址为DATA的100D字数组中的最小偶数,并把它放在AX中。

此题根据题意要求应该使用完整汇编格式,因为精简格式也使用DATA....
dataare segment
DATA DW 3322H,4555H,200H,3544H,1245H
num DB ($-DATA)/2
dataare ends
code segment
assume cs:code,ds:dataare
MAIN PROC FAR
start:
push ds
mov ax,0H
push ax
mov ax,dataare
mov ds,ax
mov dx,0H ;偶数数量计数
mov cx,Word ptr [num]
mov si,0H
isO:
mov bx,WORD PTR DATA[si]
shr bx,1H
jnc addDX
jmp commonControl
addDX:
ADD dx,1H
cmp dx,1H
je firstAXValue
compare:
mov bx,WORD PTR DATA[si]
cmp bx,ax
jge commonControl
mov ax,bx
commonControl:
ADD si,2H
loop isO
jmp exit
firstAXvalue:
mov ax,WORD PTR DATA[si]
jmp compare
exit:
ret
MAIN ENDP
code ends
END start
5.8没有做。

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

.model small
.386
.code
main proc far
start:
push ds
mov ax,0
push ax
mov ax,0
mov cx,4
mov dx,0
input:
mov ah,1H
int 21H
cmp al,30H
jl input
cmp al,39H
jle s
cmp al,61H
jl input
cmp al,66H
jg input
sub al,27H
s:
sub al,30H
add dl,al
cmp cx,1H
je f
rol dx,4
f:
loop input
exit:
call print
ret
main endp
print proc near
mov cx,10H
p:
rol dx,1
push dx
and dx,1
add dx,30H
mov ah,2
int 21H
pop dx
loop p
ret
print endp
end start
5.10设有一段英文,其变量字符名为ENG,并以$字符结束。

试编写一程序,查对单词SUN在该文中的出现次数,并以格式"SUN XXXX"显示出次数。

.model small
.386
.data
ENG db 'SUN's SUN JAVA',24H ;;;;24H = $ MESSAGE db 'SUN:','$'
.stack 100H
.code
MAIN PROC FAR
start:
PUSH DS
AND AX,0
PUSH AX。

相关文档
最新文档