1、CODES SEGMENT ASSUME CS:CODES,DS:DATAS,SS:STACKSSTART: MOV AX,DATAS MOV DS,AX;把datas的段地址分别给DS,ES mov es,ax lea si,string_a;取string_a的偏移地址给si lea di,string_b;取string_b的偏移地址给di mov cx,count;把字符串长度给cx,决定循环的次数 Cld;DF置零,使DI,SI自增rep movsb;把string_a里的字符串复制到string_b,循环次数为CX把count赋给CX,决定输出正序循环次数 lea dx,data1
2、;输出data1里的字符 mov ah,09 int 21h xor di,di;DI置零output1: mov dl,string_b+di;把string_b偏移地址给DL mov ah,02h int 21h;输出数据段string_b里的一个字符 inc di;DI自增,指向下一个字节loop output1;循环输出CX重新获值 lea dx,data2;把data2的偏移地址赋给DX int 21h ;输出提反序输出的提示字符output2: dec di;上一次循环DI指向要输出字符串的下一个字节,先自减,使他指向最后一个字符 mov dl,string_b+di mov ah
3、,02输出反序的字符串 loop output2 MOV AH,4CH INT 21H;结束dosCODES ENDSEND START;程序结束运行结果:过程中有几次语法错误,比如直接 MOV DI,STRING_B,提示错误,不能直接把数据段的偏移地址赋给DI,应该用LEA。XX得知:LEA为硬指令,在程序执行阶段得出偏移地址,offset是伪指令,在编译阶段得出偏移地址。论速度的话,offset比LEA快得多,但是,灵活性和复杂度,offset远不如LEA。2、利用中断调用,在屏幕上显示1-9中的随机数,中断号为86HRandom子程序是上网XX的,加以调用刚开始运行是这样的结果发生除法
4、溢出错误,检查后发现是AND写成了ADD改正后原子程序如下:结果很匪夷所思.是D;检查了一下,发现.我写错了,明明子程序中都把随机数放在BL里保存,而我写的output子程序里,是把AH的值赋给DL,AH的值在POP AX已经改变.最后改正后,随机数出炉了,先看程序: MOV DS,AX mov ax,0 mov es,ax;段偏移地址指向0000 mov bx,86h*4;把中断号86H的地址赋给BX mov ax,offset randommov es:word ptr bx,ax;把中断子程序的偏移地址存入ES:BX mov ax,seg random;取中断子程序的段基地址 mov e
5、s:word ptr bx+2,ax;把段基地址存入ES:BX+2 int 86h;调用中断向量表的86号即RANDOM子程序 call output;调用output子程序结束random proc PUSH CX PUSH DX PUSH AX;寄存器保护 STI;开中断,后面要调用int 1ah MOV AH,0 INT 1AH;读取时钟滴答计数 MOV AX,DX;把随机数存入AX and ax,15 MOV DL,10 DIV DL;除以10,获得1-9的随机数 MOV BL,AH ;把随机数存入AH POP AX POP DX POP CX;恢复寄存器的值 IRETrandom E
6、NDP;子程序返回 output proc near add bl,0 mov dl,bl Ret;很简单的输出子程序 output endpEND START运行结果因为是随机的,所以就不一一运行了结束!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】 data1 db please input 10 number:提示输入please input next提示输入下一个成绩 score6 db score6:60
7、-69= score7 db score7:70-79= score8 db score8:80-89= score9 db score9:90-99= score10 db score10:100=显示类别 buf db 50 db ? db 50 dup(?) ;定义缓冲区大小 score db 5 dup(0) ;暂时存放一个输入的成绩,等待运算score1 db 5 dup(0) ;存放成绩类别的个数,第一个存放的是60-69的个数,以此类推 db 100 dup(0) lea dx,data1 mov ah,09h输出提示输入的字符串 mov dl,10换行 mov cx,10;置循
8、环次数s: push cx;保护CX call input;调用input子程序 call datacate;调用datacate子程序 pop cx;还原CX loop s;循环输入调用输出子程序 mov ah,4ch int 21h ;结束 input proc near;输入子程序开始 lea dx,buf mov ah,0Ah输入第一个成绩存入缓冲区 xor ch,ch mov cl, buf+1;实际输入的字节数存入CL retinput endpdatacate proc near;处理子程序开始 cmp cx,3;把CX和3比较,等于3,即输入的是100,调转到S10 je s1
9、0 dec cx;CX自减,下面的循环次数刚好使成绩全部进入SCORE lea si,buf+2 xor di,di tt1: mov al,si mov score+di,al inc si把输入的成绩存放到score中 loop tt1 sub al,30h;al里存放的是十位数的ASCII代码,减去30h等于十进制形式 cmp al,6;和6比较 je s6;等于则跳转到s6,下面类似 cmp al,7 je s7 cmp al,8 je s8 cmp al,9 je s9s6: mov al,1;使个数能加一 add score1,al;60-69的类别人数加一,下面以此类推 jmp
10、ss1 ;跳转到提示输入下一个成绩的SS1中 s7: mov al,1 add score1+1,al jmp ss1s8: add score1+2,al s9: add score1+3,al jmp ss1 s10: add score1+4,alss1: lea dx,data2换行 datacate endp ;子程序结束output proc mov bl,30h lea dx,score6;把60-69人数转换成ASCII输出score6:60-69= add bl,score1输出人数,以下类似 lea dx,score7 add bl,score1+1 lea dx,scor
11、e8 add bl,score1+2 lea dx,score9 add bl,score1+3 lea dx,score10 add bl,score1+4主程序结束运行成功!程序写的过程中遇到的错误:1.要用到的寄存器太多,有涉及程序调用,在调用input和datacata子程序时,没有注意保护CX,是程序陷入了死循环,不断地输入输入2.我本来是没有弄这个缓冲区的而是写的SCORE2 DB 50 DUP(0)普通的数据段,发现输入一个成绩后就光标不动了,mov ah,0a int 21h是向缓冲区输入数值的。 心得 两次实验写了七个程序(准确点),十六进制转换实验刚做的时候一筹莫展,那时学
12、到的指令比较少,掌握的知识比较少,大多是用到就查,对它们的深层含义掌握的比较少。刚开始是通常是上网XX一个类似的程序。理解后再自己写。这次可以自己写了,有点又臭又长的感觉。 本来以为汇编很简单的,但是条条框框太多,字节字位这类的运算老是会产生错误,还要担心运算影响标志位,像上次实验数据相加没有进位,这次实验DF的值影响DI的变化,以为掌握了就好了,可是掌握又不太容易,书看了一遍,用的时候又记不住,比较模糊,一旦运行才知道错了,也幸亏我写的程序不太长,我检查检查还能发现错误。不过通过这几次实验,我真心觉得程序在于写,记那些指令什么的没啥用,看程序也有用,掌握不同的方法,精简自己的程序。 第四次实验有一个系统调用系统时钟产生随机数,真是想不到啊,这就需要我们多看程序。 好了,以上是我汇编的心得。
copyright@ 2008-2023 冰点文库 网站版权所有
经营许可证编号:鄂ICP备19020893号-2