汇编程序编写提高.docx

上传人:b****5 文档编号:12215772 上传时间:2023-04-17 格式:DOCX 页数:16 大小:122.20KB
下载 相关 举报
汇编程序编写提高.docx_第1页
第1页 / 共16页
汇编程序编写提高.docx_第2页
第2页 / 共16页
汇编程序编写提高.docx_第3页
第3页 / 共16页
汇编程序编写提高.docx_第4页
第4页 / 共16页
汇编程序编写提高.docx_第5页
第5页 / 共16页
点击查看更多>>
下载资源
资源描述

汇编程序编写提高.docx

《汇编程序编写提高.docx》由会员分享,可在线阅读,更多相关《汇编程序编写提高.docx(16页珍藏版)》请在冰豆网上搜索。

汇编程序编写提高.docx

汇编程序编写提高

第四次实验

汇编程序编写和提高

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的变化,以为掌握了就好了,可是掌握又不太容易,书看了一遍,用的时候又记不住,比较模糊,一旦运行才知道错了,也幸亏我写的程序不太长,我检查检查还能发现错误。

不过通过这几次实验,我真心觉得程序在于写,记那些指令什么的没啥用,看程序也有用,掌握不同的方法,精简自己的程序。

第四次实验有一个系统调用系统时钟产生随机数,真是想不到啊,这就需要我们多看程序。

好了,以上是我汇编的心得。

 

 

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

当前位置:首页 > 表格模板 > 合同协议

copyright@ 2008-2022 冰豆网网站版权所有

经营许可证编号:鄂ICP备2022015515号-1