1632位微机原理第3章课后习题答案.docx

上传人:b****8 文档编号:9418549 上传时间:2023-05-19 格式:DOCX 页数:28 大小:24.11KB
下载 相关 举报
1632位微机原理第3章课后习题答案.docx_第1页
第1页 / 共28页
1632位微机原理第3章课后习题答案.docx_第2页
第2页 / 共28页
1632位微机原理第3章课后习题答案.docx_第3页
第3页 / 共28页
1632位微机原理第3章课后习题答案.docx_第4页
第4页 / 共28页
1632位微机原理第3章课后习题答案.docx_第5页
第5页 / 共28页
1632位微机原理第3章课后习题答案.docx_第6页
第6页 / 共28页
1632位微机原理第3章课后习题答案.docx_第7页
第7页 / 共28页
1632位微机原理第3章课后习题答案.docx_第8页
第8页 / 共28页
1632位微机原理第3章课后习题答案.docx_第9页
第9页 / 共28页
1632位微机原理第3章课后习题答案.docx_第10页
第10页 / 共28页
1632位微机原理第3章课后习题答案.docx_第11页
第11页 / 共28页
1632位微机原理第3章课后习题答案.docx_第12页
第12页 / 共28页
1632位微机原理第3章课后习题答案.docx_第13页
第13页 / 共28页
1632位微机原理第3章课后习题答案.docx_第14页
第14页 / 共28页
1632位微机原理第3章课后习题答案.docx_第15页
第15页 / 共28页
1632位微机原理第3章课后习题答案.docx_第16页
第16页 / 共28页
1632位微机原理第3章课后习题答案.docx_第17页
第17页 / 共28页
1632位微机原理第3章课后习题答案.docx_第18页
第18页 / 共28页
1632位微机原理第3章课后习题答案.docx_第19页
第19页 / 共28页
1632位微机原理第3章课后习题答案.docx_第20页
第20页 / 共28页
亲,该文档总共28页,到这儿已超出免费预览范围,如果喜欢就下载吧!
下载资源
资源描述

1632位微机原理第3章课后习题答案.docx

《1632位微机原理第3章课后习题答案.docx》由会员分享,可在线阅读,更多相关《1632位微机原理第3章课后习题答案.docx(28页珍藏版)》请在冰点文库上搜索。

1632位微机原理第3章课后习题答案.docx

1632位微机原理第3章课后习题答案

 

第3章

3.1:

汇编语言是一种以处理器指令系统为基础的低级程序设计语言,它采用助记符表达指令操作码,采用标识符号表示指令操作数,可以直接、有效地控制计算机硬件,因而容易创建代码序列短小、运行快速的可执行程序

3.2解:

(1)完整的汇编语言源程序由段组成

(2)一个汇编语言源程序可以包含若干个代码段、数据段、附加段或堆栈段,段与段之间的顺序可随意排列

(3)需独立运行的程序必须包含一个代码段,并指示程序执行的起始点,一个程序只有一个起始点

(4)所有的可执行性语句必须位于某一个代码段内,说明性语句可根据需要位于任一段内

(5)通常,程序还需要一个堆栈段

3.3解:

存储模式

特点

TINY

COM类型程序,只有一个小于64KB的逻辑段(MASM6.x支持)

SMALL

小应用程序,只有一个代码段和一个数据段(含堆栈段),每段不大于64KB

COMPACT

代码少、数据多的程序,只有一个代码段,但有多个数据段

MEDIUM

代码多、数据少的程序,可有多个代码段,只有一个数据段

LARGE

大应用程序,可有多个代码段和多个数据段(静态数据小于64KB)

HUGE

更大应用程序,可有多个代码段和多个数据段(对静态数据没有限制)

FLAT

32位应用程序,运行在32位80x86CPU和Windows9x或NT环境

3.4解:

开始位置:

用标号指明

返回DOS:

利用DOS功能调用的4CH子功能来实现

汇编停止:

执行到一条END伪指令时,停止汇编

3.5解:

段定位、段组合和段类型。

3.6给出采用一个源程序格式书写的例题3.1源程序

例题3.1:

创建一个在屏幕上显示一段信息的程序……解:

stacksegmentstack

db1024(0)

stackends

datasegment

stringdb'Hello,Assembly!

',0dH,0aH,‘$’

dataends

codesegment'code'

assumecs:

code,ds:

data,ss:

stack

start:

movdx,offsetstring

movah,9

int21h

codeends

endstart

3.7DOS支持哪两种可执行程序结构,编写这两种程序时需要注意什么?

解:

(1).EXE程序

程序可以有多个代码段和多个数据段,程序长度可以超过64KB

通常生成EXE结构的可执行程序

(2).COM程序

只有一个逻辑段,程序长度不超过64KB

需要满足一定条件才能生成COM结构的可执行程序(MASM6.x需要采用TINY模式)

3.8举例说明等价“EUQ”伪指令和等号“=”伪指令的用途

解:

符号定义伪指令有“等价EQU”和“等号=”:

符号名EQU数值表达式

符号名EQU<字符串>

符号名=数值表达式

EQU用于数值等价时不能重复定义符号名,但“=”允许有重复赋值。

例如:

X=7;等效于:

Xequ7

X=X+5;“XEQUX+5”是错误的

3.9解:

(1)al=67h

(2)ax=133h,dx=4h

(3)ax=0230h

(4)al=41h

(4)ax=7654h

3.10解:

(1)

41h

42h

43h

10

10h

45h

46h

-1

?

4

4

4

-1

4

4

4

-1

4

4

4

(2)

10h

00h

0fbh

0ffh

?

?

?

?

?

?

3.11

解:

.data

my1bdb'PersonalComputer'

my2bdb20

my3bdb14h

my4bdb00010100b

my5wdw20dup(?

my6c=100

my7c=<'PersonalComputer'>

3.12希望控制变量或程序代码在段中的偏移地址,应该使用哪个伪指令?

解:

利用定位伪指令控制,如org,even,align

3.13名字和符号有什么属性?

解:

包括逻辑地址和类型两种属性。

3.14解:

;数据段

org100h

varwdw1234h,5678h

varbdb3,4

varddd12345678h

buffdd10dup(?

messdb'Hello'

;代码段

movax,offsetvarb+offsetmess

movax,typebuff+typemess+typevard

movax,sizeofvarw+sizeofbuff+sizeofmess

movax,lengthofvarw+lengthofvard

3.15解:

(1)1000超过一个字节所能表达的最大整数

(2)SI应为偶数

(3)两个内存单元不能直接运算

(4)应改为[al+1]

(5)条件转移指令后面应接标号,而不是变量

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

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

解:

(1)汇编语言中,子程序要用一对过程伪指令PROC和ENDP声明,格式如下:

过程名PROC[NEAR|FAR]

……;过程体

过程名ENDP

(2)保护用到的寄存器内容,以便子程序返回时进行相应的恢复。

(3)改错:

crazyproc

pishbx

pushcx

xorax,ax

xordx,dx

again:

adda,[bx]

adcdx,0

incbx

incbx

loopagain

popcx

popbx

3.27编写一个源程序,在键盘上按一个键,将从AL返回的ASCⅡ码值显示出来,如果按下ESC键则程序退出。

解(不需调用HTOASC子程序):

again:

movah,1

int21h

cmpal,1bh;ESC的ASCII码是1bh

jedone

movdl,al

movah,2

int21h;是大写字母则转换为小写字母

jmpagain

done:

3.28

解答:

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调用HTOASC子程序,显示一个字节的16进制数,后跟“H”的子程序。

解:

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写一个子程序,根据入口参数AL=0、1、2,依次实现对大写字母转换成小写、小写转换成大写或大小字母互换。

欲转换的字符串在string中,用0表示结束

解:

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.31子程序的参数传递有哪些方法?

请简单比较

解:

(1)用寄存器传递参数:

最简单和常用的参数传递方法是通过寄存器,只要把参数存于约定的寄存器中就可以了

由于通用寄存器个数有限,这种方法对少量数据可以直接传递数值,而对大量数据只能传递地址

采用寄存器传递参数,注意带有出口参数的寄存器不能保护和恢复,带有入口参数的寄存器可以保护、也可以不保护,但最好能够保持一致

(2)用共享变量传递参数

子程序和主程序使用同一个变量名存取数据就是利用共享变量(全局变量)进行参数传递

如果变量定义和使用不在同一个源程序中,需要利用PUBLIC、EXTREN声明

如果主程序还要利用原来的变量值,则需要保护和恢复

利用共享变量传递参数,子程序的通用性较差,但特别适合在多个程序段间、尤其在不同的程序模块间传递数据

(3)用堆栈传递参数

参数传递还可以通过堆栈这个临时存储区。

主程序将入口参数压入堆栈,子程序从堆栈中取出参数;子程序将出口参数压入堆栈,主程序弹出堆栈取得它们

采用堆栈传递参数是程式化的,它是编译程序处理参数传递、以及汇编语言与高级语言混合编程时的常规方法

3.32采用堆栈传递参数的一般方法是什么?

为什么应该特别注意堆栈平衡问题。

解:

方法:

主程序将入口参数压入堆栈,子程序从堆栈中取出参数;子程序将出口参数压入堆栈,主程序弹出堆栈取得它们

注意:

压栈与弹栈必须要一一对应。

3.33编写一个求32位数据补码的子程序,通过寄存器传递入口参数

解:

方法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

展开阅读全文
相关资源
猜你喜欢
相关搜索

当前位置:首页 > IT计算机 > 电脑基础知识

copyright@ 2008-2023 冰点文库 网站版权所有

经营许可证编号:鄂ICP备19020893号-2