汇编程序编写提高文档格式.docx
《汇编程序编写提高文档格式.docx》由会员分享,可在线阅读,更多相关《汇编程序编写提高文档格式.docx(16页珍藏版)》请在冰点文库上搜索。
![汇编程序编写提高文档格式.docx](https://file1.bingdoc.com/fileroot1/2023-5/3/80666e34-1c98-41a9-8685-0d1b1ca7040d/80666e34-1c98-41a9-8685-0d1b1ca7040d1.gif)
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
把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;
循环输出
CX重新获值
leadx,data2;
把data2的偏移地址赋给DX
int21h;
输出提反序输出的提示字符
output2:
decdi;
上一次循环DI指向要输出字符串的下一个字节,先自减,
使他指向最后一个字符
movdl,[string_b+di]
movah,02
输出反序的字符串
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已经改变......最后改正后,随机数出炉了,先看程序:
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子程序
结束
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
Ret;
很简单的输出子程序
outputendp
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】
data1db'
pleaseinput10number:
提示输入
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的个数,
以此类推
db100dup(0)
leadx,data1
movah,09h
输出提示输入的字符串
movdl,10
换行
movcx,10;
置循环次数
s:
pushcx;
保护CX
callinput;
调用input子程序
calldatacate;
调用datacate子程序
popcx;
还原CX
loops;
循环输入
调用输出子程序
movah,4ch
int21h;
结束
inputprocnear;
输入子程序开始
leadx,buf
movah,0Ah
输入第一个成绩存入缓冲区
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
把输入的成绩存放到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:
add[score1+2],al
s9:
add[score1+3],al
jmpss1
s10:
add[score1+4],al
ss1:
leadx,data2
换行
datacateendp;
子程序结束
outputproc
movbl,30h
leadx,score6;
把60-69人数转换成ASCII
输出‘score6:
60-69=’
addbl,[score1]
输出人数,以下类似
leadx,score7
addbl,[score1+1]
leadx,score8
addbl,[score1+2]
leadx,score9
addbl,[score1+3]
leadx,score10
addbl,[score1+4]
主程序结束
运行成功!
程序写的过程中遇到的错误:
1.要用到的寄存器太多,有涉及程序调用,在调用input和datacata子程序时,没有注意保护CX,是程序陷入了死循环,不断地输入输入
2.我本来是没有弄这个缓冲区的
而是写的SCORE2DB50DUP(0)普通的数据段,发现输入一个成绩后就光标不动了,movah,0aint21h是向缓冲区输入数值的。
心得
两次实验写了七个程序(准确点),十六进制转换实验刚做的时候一筹莫展,那时学到的指令比较少,掌握的知识比较少,大多是用到就查,对它们的深层含义掌握的比较少。
刚开始是通常是上网XX一个类似的程序。
理解后再自己写。
这次可以自己写了,有点又臭又长的感觉。
本来以为汇编很简单的,但是条条框框太多,字节字位这类的运算老是会产生错误,还要担心运算影响标志位,像上次实验数据相加没有进位,这次实验DF的值影响DI的变化,以为掌握了就好了,可是掌握又不太容易,书看了一遍,用的时候又记不住,比较模糊,一旦运行才知道错了,也幸亏我写的程序不太长,我检查检查还能发现错误。
不过通过这几次实验,我真心觉得程序在于写,记那些指令什么的没啥用,看程序也有用,掌握不同的方法,精简自己的程序。
第四次实验有一个系统调用系统时钟产生随机数,真是想不到啊,这就需要我们多看程序。
好了,以上是我汇编的心得。