1632位微机原理汇编语言及接口技术第三章习题答案.docx
《1632位微机原理汇编语言及接口技术第三章习题答案.docx》由会员分享,可在线阅读,更多相关《1632位微机原理汇编语言及接口技术第三章习题答案.docx(20页珍藏版)》请在冰点文库上搜索。
1632位微机原理汇编语言及接口技术第三章习题答案
第3章汇编语言程序设计(习题3)
3.16解:
movah,1;只允许输入小写
int21h
subal,20h;转换为大写字母
movdl,al
movah,2
int21h;显示
3.17解:
movbx,offsetLEDtable
moval,lednum
xlat
3.18解:
movax,bufX
cmpax,bufY
jaedone
movax,bufY
done:
movbufZ,ax
3.19解:
.modelsmall
.stack
.data
bufXdw-7
signXdb?
.code
.startup
cmpbufX,0;testbufX,80h
jlnext;jnznext
movsignX,0
jmpdone
next:
movsignX,-1
done:
.exit0
end
3.20解:
movdl,’2’
movax,bufX
cmpax,bufY
jenext1
decdl
next1:
cmpax,bufZ
jenext2
decdl
next2:
movah,2
int21h
3.21解:
;代码段
moval,number
movbx,0;BX←记录为1的位数
restart:
cmpal,0;AL=0结束
jzdone
again:
shral,1;最低位右移进入CF
jcnext;为1,转移
incbx;不为1,继续
jmpagain
next:
pushax
pushbx
shlbx,1;位数乘以2(偏移地址要用2个字节单元)
jmpaddrs[bx];间接转移:
IP←[table+BX]
;以下是各个处理程序段
fun0:
movdl,'0'
jmpdisp
fun1:
movdl,'1'
jmpdisp
fun2:
movdl,'2'
jmpdisp
fun3:
movdl,'3'
jmpdisp
fun4:
movdl,'4'
jmpdisp
fun5:
movdl,'5'
jmpdisp
fun6:
movdl,'6'
jmpdisp
fun7:
movdl,'7'
jmpdisp
;
disp:
movah,2;显示一个字符
int21h
popbx
popax
jmprestart
done:
…
3.22编制程序完成12H、45H、0F3H、6AH、20H、0FEH、90H、0C8H、57H和34H等10个字节数据之和,并将结果存入字节变量SUM中(不考虑溢出和进位)。
;wjxt322.asm
.modelsmall
.stack
.data
b_datadb12h,45h,0f3h,6ah,20h,0feh,90h,0c8h,57h,34h;原始数据
numequ10;数据个数
sumdb?
;预留结果单元
.code
.startup
xorsi,si;位移量清零
xoral,al;取第一个数
movcx,num;累加次数
again:
addal,b_data[si];累加
incsi;指向下一个数
loopagain;如未完,继续累加
movsum,al;完了,存结果
.exit0
end
3.23求主存0040h:
0开始的一个64KB物理段中共有多少个空格?
;wjxt323.asm
.modelsmall
.code
start:
movax,0040h;送段地址
movds,ax
movsi,0;偏移地址
movcx,si;计数(循环次数)
xorax,ax;空格计数器清零
again:
cmpbyteptr[si],20h;与空格的ASCII码比较
jnenext;不是空格,转
incax;是空格,空格数加1
next:
incsi;修改地址指针
loopagain;cx=cx-1,如cx=0退出循环
.exit0
endstart
3.24编写计算100个16位正整数之和的程序。
如果和不超过16位字的范围(65535),则保存其和到wordsum,如超过则显示‘overflow’。
答:
;数据段
countequ100
parraydwcountdup(?
);假设有100个数据
wordsumdw0
msgdb‘overflow’,’$’
;代码段
movcx,count
movax,0
movbx,offsetparray
again:
addax,[bx]
jncnext
movdx,offsetmsg
movah,9
int21h;显示溢出信息
jmpdone;然后,跳出循环体
next:
addbx,2
loopagain
movwordsum,ax
done:
…
3.25编程把—个16位无符号二进制数转换成为用8421BCD码表示的5位十进制数。
转换算法可以是:
用二进制数除以10000,商为“万位”,再用余数除以1000,得到“千位”;依次用余数除以l00、10和l,得到“百位”、“十位”和“个位”。
;wjxt325.asm
.modelsmall
.stack256
.data
arraydw?
;源字数据
dbcddb5dup(?
);五位bcd结果,高对高低对低
.code
.startup
movdx,array;取源数据(余数)
movbx,10000;除数
movcx,10;除数系数
movsi,4;目的数据高位位移量
again:
movax,dx;dx.ax中存放被除数
movdx,0
divbx;除于bx,商ax,余数dx
movdbcd[si],al;商<10,存结果
pushdx;暂存余数
movax,bx;除数除于10
movdx,0
divcx;dx.ax除于cx,商ax、余数0存在dx
movbx,ax;bx是除数
popdx
decsi;目的数据位移量减1
jnzagain
movdbcd,dl;存个位数(<10)
.exit0
end
3.27解(不需调用HTOASC子程序):
again:
movah,1
int21h
cmpal,1bh;ESC的ASCII码是1bh
jedone
movdl,al
movah,2
int21h;是大写字母则转换为小写字母
jmpagain
done:
…
3.28请按如下说明编写子程序:
;子程序功能:
把用ASCII码表示的两位十进制数转换为对应二进制数
;入口参数:
DH=十位数的ASCII码,DL=个位数的ASCII码
;出口参数:
AL=对应的二进制数
答:
asctobproc
pushcx
anddh,0fh;先转换十位数
shldh,1;十位数乘以10(采用移位指令)
movch,dh
shldh,1
shldh,1
adddh,ch
anddl,0fh;转换个位数
adddh,dl;十位数加个位数
moval,dh;设置出口参数
popcx
ret
asctobendp
3.29解:
DIPASCproc;入口参数:
AL=要显示的一个16进制数
pushcx
pushdx
pushax
movcl,4;转换高位
shral,cl
callHTOASC
movdl,al;显示
movah,2
int21h
popax;转换低位
callHTOASC
movdl,al;显示
movah,2
int21h
movdl,’H’;显示一个字母“H”
movah,2
int21h
popdx
popcx
ret
DIPASCendp
HTOASCproc;将AL低4位表达的一位16进制数转换为ASCII码
andal,0fh
cmpal,9
jbehtoasc1
addal,37h;是0AH~0FH,加37H转换为ASCII码
ret;子程序返回
htoasc1:
addal,30h;是0~9,加30H转换为ASCII码
ret;子程序返回
HTOASCendp
3.30解:
lucaseproc
pushbx
movbx,offsetstring
cmpal,0
jecase0
cmpal,1
jzcase1
cmpal,2
jzcase2
jmpdone
case0:
cmpbyteptr[bx],0
jedone
cmpbyteptr[bx],’A’
jbnext0
cmpbyteptr[bx],’Z’
janext0
addbyteptr[bx],20h
next0:
incbx
jmpcase0
case1:
cmpbyteptr[bx],0
jedone
cmpbyteptr[bx],’a’
jbnext1
cmpbyteptr[bx],’z’
janext1
subbyteptr[bx],20h
next1:
incbx
jmpcase1
case2:
cmpbyteptr[bx],0
jedone
cmpbyteptr[bx],’A’
jbnext2
cmpbyteptr[bx],’Z’
janext20
addbyteptr[bx],20h
jmpnext2
next20:
cmpbyteptr[bx],’a’
jbnext2
cmpbyteptr[bx],’z’
janext2
subbyteptr[bx],20h
next2:
incbx
jmpcase2
done:
popbx
ret
lucaseendp
3.33解:
方法1:
neg32proc;入口参数:
DX.AX=32位有符号数
negax;实现0-DX.AX功能
negdx
sbbdx,0;这条指令也可以用decdx代替
ret
neg32endp;出口参数:
DX.AX=32位有符号数的补码
方法2:
neg32proc;入口参数:
DX.AX=32位有符号数
notax;实现DX.AX求反加1
notdx
addax,1
adcdx,0
ret
neg32endp;出口参数:
DX.AX=32位有符号数的补码
3.34解:
;数据段
arraydb12h,25h,0f0h,0a3h,3,68h,71h,0cah,0ffh,90h;数组
countequ$-array;数组元素个数
resultdb?
;校验和
;代码段
movbx,offsetarray;BX←数组的偏移地址
movcx,count;CX←数组的元素个数
callchecksum;调用求和过程
movresult,al;处理出口参数
movax,4c00h
int21h
;计算字节校验和的通用过程
;入口参数:
DS:
BX=数组的段地址:
偏移地址,CX=元素个数
;出口参数:
AL=校验和
;说明:
除AX/BX/CX外,不影响其他寄存器
checksumproc
xoral,al;累加器清0
sum:
addal,[bx];求和
incbx;指向下一个字节
loopsum
ret
checksumendp
end
3.35解:
⑴
.modelsmall
.stack
.data
wdatadw34abh
.code
.startup
movax,wdata
calldispa
.exit0
;
dispaproc
pushcx
pushdx
movcl,4
movdl,ah
shrdl,cl
calldldisp
movdl,ah
anddl,0fh
calldldisp
movdl,al
shrdl,cl
calldldisp
movdl,al
anddl,0fh
calldldisp
popdx
popcx
ret
dispaendp
;
dldispproc
pushax
ordl,30h
cmpdl,39h
jbedldisp1
adddl,7
dldisp1:
movah,2
int21h
popax
ret
dldispendp
end
⑵
.modelsmall
.stack
.data
wdatadw34abh
wordtempdw?
.code
.startup
movax,wdata
movwordtemp,ax
calldispa
.exit0
;
dispaproc
pushcx
pushdx
movcl,4
movdl,byteptrwordtemp+1
shrdl,cl
calldldisp
movdl,byteptrwordtemp+1
anddl,0fh
calldldisp
movdl,byteptrwordtemp
shrdl,cl
calldldisp
movdl,byteptrwordtemp
anddl,0fh
calldldisp
popdx
popcx
ret
dispaendp
;
dldispproc
pushax
ordl,30h
cmpdl,39h
jbedldisp1
adddl,7
dldisp1:
movah,2
int21h
popax
ret
dldispendp
end
⑶
.modelsmall
.stack
.data
wdatadw34abh
.code
.startup
pushwdata
calldispa
popax;addsp,2
.exit0
;
dispaproc
pushbp
movbp,sp
pushax
pushcx
pushdx
movax,[bp+4]
movcl,4
movdl,ah
shrdl,cl
calldldisp
movdl,ah
anddl,0fh
calldldisp
movdl,al
shrdl,cl
calldldisp
movdl,al
anddl,0fh
calldldisp
popdx
popcx
popax
popbp
ret
dispaendp
;
dldispproc
pushax
ordl,30h
cmpdl,39h
jbedldisp1
adddl,7
dldisp1:
movah,2
int21h
popax
ret
dldispendp
end
3.40编写一个宏指令movedoprnd,soprnd,它实现任意寻址方式的字量源操作数soprnd送到目的操作数doprnd,包括存储单元到存储单元的传送功能。
答:
movemacrodoprnd,soprnd
movax,soprnd
movdoprnd,ax
endm
3.41定义一个宏logical,用它代表4条逻辑运算指令:
and/or/xor/test。
注意需要利用3个形式参数,并给出一个宏调用以及对应宏展开的例子。
答:
logicalmacrolcode,dopd,sopd
lcodedopd,sopd
endm
例如,如果使用“andax,[bx]”指令,可以利用该宏定义,写出宏指令如下:
logicaland,ax,[bx]
3.42解:
utolmacro
localnext
cmpal,’A’;小于“A”不转换
jbnext
cmpal,’Z’;大于“A”不转换
janext
addal,20h;是大写字母则转换为小写字母
next:
endm
3.43定义一个宏movestrstrn,dstr,sstr,它将strn个字符从一个字符区sstr传送到另一个字符区dstr
解:
(假设它们都在数据段)
movestrmacrostrn,dstr,sstr
movcx,ds
moves,cx
movcx,strn
movdi,offsetdstr
movsi,offsetsstr
cld
repmovsb;;重复传送ES:
[DI]←DS:
[SI]
endm