汇编程序编写提高.docx
- 文档编号:12215772
- 上传时间:2023-04-17
- 格式:DOCX
- 页数:16
- 大小:122.20KB
汇编程序编写提高.docx
《汇编程序编写提高.docx》由会员分享,可在线阅读,更多相关《汇编程序编写提高.docx(16页珍藏版)》请在冰豆网上搜索。
汇编程序编写提高
第四次实验
汇编程序编写和提高
1、编写一个程序,实现复制字符串的功能,并将复制的字符串正序倒序显示出来。
流程图:
程序如下:
DATASSEGMENT
string_adb'thefutureisbeautiful';要操作的字符串
countequ$-offsetstring_a;把字符串的长度存入count
data1db'thezhengxuis:
','$';输出正序是
data2db'thefanxuis:
','$';输出反序是
DATASENDS
STACKSSEGMENT
string_bdb100dup(?
);建栈段存放复制的字符串
STACKSENDS
CODESSEGMENT
ASSUMECS:
CODES,DS:
DATAS,SS:
STACKS
START:
MOVAX,DATAS
MOVDS,AX;把datas的段地址分别给DS,ES
moves,ax
leasi,string_a;取string_a的偏移地址给si
leadi,string_b;取string_b的偏移地址给di
movcx,count;把字符串长度给cx,决定循环的次数
Cld;DF置零,使DI,SI自增
repmovsb;把string_a里的字符串复制到string_b,循环次数为
;CX
movcx,count;把count赋给CX,决定输出正序循环次数
leadx,data1;输出data1里的字符
movah,09
int21h
xordi,di;DI置零
output1:
movdl,[string_b+di];把string_b偏移地址给DL
movah,02h
int21h;输出数据段string_b里的一个字符
incdi;DI自增,指向下一个字节
loopoutput1;循环输出
movcx,count;CX重新获值
leadx,data2;把data2的偏移地址赋给DX
movah,09
int21h;输出提反序输出的提示字符
output2:
decdi;上一次循环DI指向要输出字符串的下一个字节,先自减,
;使他指向最后一个字符
movdl,[string_b+di]
movah,02
int21h;输出反序的字符串
loopoutput2
MOVAH,4CH
INT21H;结束dos
CODESENDS
ENDSTART;程序结束
运行结果:
过程中有几次语法错误,比如直接MOVDI,STRING_B,提示错误,不能直接把数据段的偏移地址赋给DI,应该用LEA。
XX得知:
LEA为硬指令,在程序执行阶段得出偏移地址,offset是伪指令,在编译阶段得出偏移地址。
论速度的话,offset比LEA快得多,但是,灵活性和复杂度,offset远不如LEA。
2、利用中断调用,在屏幕上显示1-9中的随机数,中断号为86H
流程图:
Random子程序是上网XX的,加以调用
刚开始运行是这样的结果
发生除法溢出错误,检查后发现是AND写成了ADD
改正后原子程序如下:
运行结果:
结果很匪夷所思......是D;检查了一下,发现....我写错了,明明子程序中都把随机数放在BL里保存,而我写的output子程序里,是把AH的值赋给DL,AH的值在POPAX已经改变......最后改正后,随机数出炉了,先看程序:
DATASSEGMENT
DATASENDS
STACKSSEGMENT
STACKSENDS
CODESSEGMENT
ASSUMECS:
CODES,DS:
DATAS,SS:
STACKS
START:
MOVAX,DATAS
MOVDS,AX
movax,0
moves,ax;段偏移地址指向0000
movbx,86h*4;把中断号86H的地址赋给BX
movax,offsetrandom
moves:
wordptr[bx],ax;把中断子程序的偏移地址存入
;ES:
[BX]
movax,segrandom;取中断子程序的段基地址
moves:
wordptr[bx+2],ax;把段基地址存入ES:
[BX+2]
int86h;调用中断向量表的86号即RANDOM子程序
calloutput;调用output子程序
MOVAH,4CH
INT21H;结束
randomproc
PUSHCX
PUSHDX
PUSHAX;寄存器保护
STI;开中断,后面要调用int1ah
MOVAH,0
INT1AH;读取时钟滴答计数
MOVAX,DX;把随机数存入AX
andax,15
MOVDL,10
DIVDL;除以10,获得1-9的随机数
MOVBL,AH;把随机数存入AH
POPAX
POPDX
POPCX;恢复寄存器的值
IRET
randomENDP;子程序返回
outputprocnear
addbl,'0'
movdl,bl
movah,02h
int21h
Ret;很简单的输出子程序
outputendp
CODESENDS
ENDSTART
运行结果
因为是随机的,所以就不一一运行了
结束!
3、键盘输入10个学生的成绩,编写一个程序统计60-69,70-79,80-89,90-99,100的人数,分别存放在score6,7,8,9,10中。
学生成绩如下【65,98,78,82,88,95,72,62,90,100】
流程图:
程序如下:
DATASSEGMENT
data1db'pleaseinput10number:
','$';提示输入
data2db'pleaseinputnext','$';提示输入下一个成绩
score6db'score6:
60-69=','$'
score7db'score7:
70-79=','$'
score8db'score8:
80-89=','$'
score9db'score9:
90-99=','$'
score10db'score10:
100=','$';显示类别
bufdb50
db?
db50dup(?
);定义缓冲区大小
scoredb5dup(0);暂时存放一个输入的成绩,等待运算
score1db5dup(0);存放成绩类别的个数,第一个存放的是60-69的个数,
;以此类推
DATASENDS
STACKSSEGMENT
db100dup(0)
STACKSENDS
CODESSEGMENT
ASSUMECS:
CODES,DS:
DATAS,SS:
STACKS
START:
MOVAX,DATAS
MOVDS,AX
leadx,data1
movah,09h
int21h;输出提示输入的字符串
movah,02
movdl,10
int21h;换行
movcx,10;置循环次数
s:
pushcx;保护CX
callinput;调用input子程序
calldatacate;调用datacate子程序
popcx;还原CX
loops;循环输入
calloutput;调用输出子程序
movah,4ch
int21h;结束
inputprocnear;输入子程序开始
leadx,buf
movah,0Ah
int21h;输入第一个成绩存入缓冲区
movah,02
movdl,10
int21h;换行
xorch,ch
movcl,buf+1;实际输入的字节数存入CL
ret
inputendp
datacateprocnear;处理子程序开始
cmpcx,3;把CX和3比较,等于3,即输入的是100,调转到S10
jes10
deccx;CX自减,下面的循环次数刚好使成绩全部进入SCORE
leasi,buf+2
xordi,di
tt1:
moval,[si]
mov[score+di],al
incsi
incdi;把输入的成绩存放到score中
looptt1
subal,30h;al里存放的是十位数的ASCII代码,减去30h等于十进制
;形式
cmpal,6;和6比较
jes6;等于则跳转到s6,下面类似
cmpal,7
jes7
cmpal,8
jes8
cmpal,9
jes9
s6:
moval,1;使个数能加一
add[score1],al;60-69的类别人数加一,下面以此类推
jmpss1;跳转到提示输入下一个成绩的SS1中
s7:
moval,1
add[score1+1],al
jmpss1
s8:
moval,1
add[score1+2],al
jmpss1
s9:
moval,1
add[score1+3],al
jmpss1
s10:
moval,1
add[score1+4],al
jmpss1
ss1:
movah,09h
leadx,data2
int21h;提示输入下一个成绩
movdl,10
movah,02
int21h;换行
ret
datacateendp;子程序结束
outputproc
movbl,30h
leadx,score6;把60-69人数转换成ASCII
movah,09
int21h;输出‘score6:
60-69=’
movah,02
addbl,[score1]
movdl,bl
int21h;输出人数,以下类似
movdl,10
int21h;换行
movbl,30h
leadx,score7
movah,09
int21h
movah,02
addbl,[score1+1]
movdl,bl
int21h
movdl,10
int21h
movbl,30h
leadx,score8
movah,09
int21h
movah,02
addbl,[score1+2]
movdl,bl
int21h
movdl,10
int21h
movbl,30h
leadx,score9
movah,09
int21h
movah,02
addbl,[score1+3]
movdl,bl
int21h
movdl,10
int21h
movbl,30h
leadx,score10
movah,09
int21h
movah,02
addbl,[score1+4]
movdl,bl
int21h
movdl,10
int21h
ret
outputendp
CODESENDS
ENDSTART;主程序结束
运行结果:
运行成功!
程序写的过程中遇到的错误:
1.要用到的寄存器太多,有涉及程序调用,在调用input和datacata子程序时,没有注意保护CX,是程序陷入了死循环,不断地输入输入
2.我本来是没有弄这个缓冲区的
而是写的SCORE2DB50DUP(0)普通的数据段,发现输入一个成绩后就光标不动了,movah,0aint21h是向缓冲区输入数值的。
心得
两次实验写了七个程序(准确点),十六进制转换实验刚做的时候一筹莫展,那时学到的指令比较少,掌握的知识比较少,大多是用到就查,对它们的深层含义掌握的比较少。
刚开始是通常是上网XX一个类似的程序。
理解后再自己写。
这次可以自己写了,有点又臭又长的感觉。
本来以为汇编很简单的,但是条条框框太多,字节字位这类的运算老是会产生错误,还要担心运算影响标志位,像上次实验数据相加没有进位,这次实验DF的值影响DI的变化,以为掌握了就好了,可是掌握又不太容易,书看了一遍,用的时候又记不住,比较模糊,一旦运行才知道错了,也幸亏我写的程序不太长,我检查检查还能发现错误。
不过通过这几次实验,我真心觉得程序在于写,记那些指令什么的没啥用,看程序也有用,掌握不同的方法,精简自己的程序。
第四次实验有一个系统调用系统时钟产生随机数,真是想不到啊,这就需要我们多看程序。
好了,以上是我汇编的心得。
- 配套讲稿:
如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
- 特殊限制:
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。
- 关 键 词:
- 汇编程序 编写 提高