您好,欢迎访问三七文档
第5章模块化程序设计5.1子程序结构5.2参数传递5.3多模块程序结构5.4宏结构第5章模块化程序设计以子程序结构为主体,围绕数码转换实现键盘输入和显示输出掌握子程序、文件包含、宏汇编等各种多模块编程的方法第5章模块化程序设计35.1子程序结构•经常用到的应用问题编写成一个通用子程序•大型处理过程分解成能够解决的模块•使用子程序可以使程序的结构更为清楚程序的维护更为方便有利于大程序开发时的多个程序员分工合作子程序(Subroutine)=函数(Function)=过程(Procedure)第5章模块化程序设计45.1.1子程序指令•子程序与主程序分开、完成特定功能的一段程序•主程序(调用程序)执行调用指令CALL调用子程序•子程序(被调用程序)执行返回指令RET返回主程序CALLlabel主程序RET子程序回到CALL指令后的指令处第5章模块化程序设计51.子程序调用指令CALL•CALL指令用在主程序中,实现子程序的调用分成段内调用(近调用)和段间调用(远调用)目标地址采用相对寻址、直接寻址或间接寻址•入栈返回地址:将CALL下条指令的地址压入堆栈CALLlabel;调用标号指定的子程序CALLreg32/reg16;调用寄存器指定地址的子程序CALLmem48/mem32/mem16;调用存储单元指定地址的子程序第5章模块化程序设计62.子程序返回指令RET•RET指令用在子程序结束,实现返回主程序RET;无参数返回:出栈返回地址RETi16;有参数返回:出栈返回地址;ESP←ESP+i16MASM会根据存储模型等信息确定子程序的远近调用,并相应产生返回指令第5章模块化程序设计73.过程定义伪指令•MASM利用过程定义伪指令获得子程序信息过程名PROC……;过程体过程名ENDP;过程名为符合语法的标识符•PROC后面可加参数:NEAR或FAR•简化段定义源程序格式中,通常不需指定第5章模块化程序设计8〔例5-1〕子程序调用程序-1;代码段,主程序0010B80001movax,10013BD0005movbp,50016E8000Dcallsubp;子程序调用0019B90003retp1:movcx,3001CBA0004retp2:movdx,4001FE80000Ecalldisprw第5章模块化程序设计9〔例5-1〕子程序调用程序-2;代码段,子程序subpproc;过程定义,过程名为subppushbpmovbp,spmovsi,[bp+2];SI=CALL下条指令(标号RETP1)偏移地址movdi,offsetretp2;DI=标号RETP2的偏移地址movbx,2popbp;弹出堆栈,保持堆栈平衡ret;子程序返回subpendp;过程结束MOV[EBP+2],DI示意图第5章模块化程序设计115.1.2子程序设计•子程序的编写方法与主程序一样•但需要留意几个问题:利用过程定义,获得子程序名和调用属性RET指令返回主程序,CALL指令调用子程序压入和弹出操作要成对使用,保持堆栈平衡开始保护寄存器,返回前相应恢复安排在代码段的主程序之外子程序允许嵌套和递归最好有完整的注释难点是参数传递第5章模块化程序设计12〔例5-2〕回车换行子程序dispcrlfproc;回车换行子程序pushax;保护寄存器pushdxmovdl,0dh;输出回车字符movah,2int21hmovdl,0ah;输出换行字符movah,2int21hpopdx;恢复寄存器popaxret;子程序返回dispcrlfendp第5章模块化程序设计135.2参数传递•主程序与子程序间通过参数传递建立联系入口参数(输入参数):主程序→子程序出口参数(输出参数):子程序→主程序•传递参数的多少反映程序模块间的耦合程度•参数的具体内容数据本身(传递数值)数据的存储地址(传递地址,传递引用)•参数传递方法寄存器变量堆栈第5章模块化程序设计145.2.1寄存器传递参数•最简单和常用的参数传递方法•把参数存于约定的寄存器少量数据直接传递数值大量数据只能传递地址•带有出口参数的寄存器不能保护和恢复•带有入口参数的寄存器可以保护、也可以不保护,但最好能够保持一致DOS功能调用未保存入口参数第5章模块化程序设计15〔例5-3〕十六进制显示程序-1moveax,1234abcdh;假设一个数据xorebx,ebxmovecx,8;8位十六进制数again:roleax,4;高4位循环移位进入低4位pusheax;movedx,eaxcallhtoasc;调用子程序movregd+4[ebx],al;保存ASCII码popeax;moveax,edxincebxloopagain……;显示regddb'EAX=',8dup(0),'H',‘$',第5章模块化程序设计16〔例5-3〕十六进制显示程序-2;子程序htoascproc;将AL低4位表达的一位十六进制数转换为ASCII码andal,0fh;只取AL的低4位oral,30h;AL高4位变成3cmpal,39h;是0~9,还是A~Fjbehtoendaddal,7;是A~F,ASCII码再加上7htoend:ret;子程序返回htoascendp第5章模块化程序设计17〔例5-3〕十六进制显示程序-3;子程序htoascprocandeax,0fh;取AL低4位moval,ASCII[eax];换码ret;子程序的局部数据(只读)ASCIIdb'0123456789ABCDEF'htoascendpEAX=1234ABCDH运行结果第5章模块化程序设计18〔例5-4〕有符号十进制数显示程序-1•转换的算法如下:1)首先判断数据是零、正数或负数,是零显示“0”退出2)是负数,显示负号“-”,求数据的绝对值3)接着数据除以10,余数为十进制数码,加30H转换为ASCII码保存4)重复第(3)步,直到商为0结束5)依次从高位开始显示各位数字第5章模块化程序设计19〔例5-4〕有符号十进制数显示程序-2;数据段arraydd1234567890,-1234,0,1,...count=lengthofarray;代码段,主程序movecx,countmovebx,0again:moveax,array[ebx*4];EAX=入口参数calldispsid;调用子程序显示一个数据incebxcalldispcrlf;换行以便显示下一个数据loopagain寄存器传递参数第5章模块化程序设计20〔例5-4〕有符号十进制数显示程序-3;显示有符号十进制数的子程序dispsidproc;EAX=入口参数pusheax;保护寄存器pushebxpushedxtesteax,eax;判断是零、正数或负数jnzdsid1;不是零,跳转movdl,‘0’;是零,显示“0”movah,2int21hjmpdsid5;转向显示寄存器传递参数第5章模块化程序设计21〔例5-4〕有符号十进制数显示程序-4dsid1:jnsdsid2;是正数,跳转movebx,eaxmovdl,'-';是负数,显示负号movah,2int21hmoveax,ebxnegeax;数据求补(绝对值)dsid2:movebx,10pushebx;10压入堆栈,作为退出标志第5章模块化程序设计22〔例5-4〕有符号十进制数显示程序-5dsid3:cmpeax,0;数据(商)为零,转向保存jzdsid4xoredx,edx;零位扩展被除数为EDX.EAXdivebx;数据除以10:EDX.EAX÷10adddl,30h;余数(0~9)转换为ASCIIpushedx;数据先低位后高位压入堆栈jmpdsid3dsid4:popedx;数据先高位后低位弹出堆栈cmpdl,10;是结束标志10,转向显示jedsid5第5章模块化程序设计23〔例5-4〕有符号十进制数显示程序-6movah,2;显示int21hjmpdsid4dsid5:popedx;恢复寄存器popebxpopeaxret;子程序返回dispsidendpdispcrlfproc;使光标回车换行的子程序……;同例题5-2,略dispcrlfendp第5章模块化程序设计245.2.2共享变量传递参数•子程序和主程序使用同一个变量名存取数据•如果变量定义和使用不在同一个程序模块中,需要利用PUBLIC、EXTREN声明•共享变量传递参数,子程序的通用性较差•特别适合在多个程序段间、尤其在不同的程序模块间传递数据第5章模块化程序设计25〔例5-5〕二进制输入程序-1;数据段count=5arrayddcountdup(0)tempdd?;共享变量;代码段,主程序movecx,countmovebx,offsetarrayagain:callreadbd;调用子程序,输入一个数据moveax,temp;获得出口参数mov[ebx],eax;存放到数据缓冲区addebx,4calldispcrlf;换行loopagain第5章模块化程序设计26〔例5-5〕二进制输入程序-2;代码段,子程序readbdproc;二进制输入子程序pusheax;出口参数:共享变量TEMPpushebxpushecxrdbd1:xorebx,ebx;EBX用于存放二进制结果movecx,32;限制输入字符的个数rdbd2:movah,1;输入一个字符int21hcmpal,'0';检测键入字符是否合法jbrderr;不合法则返回重新输入cmpal,'1'jarderr第5章模块化程序设计27〔例5-5〕二进制输入程序-3subal,'0';对输入的字符进行转化shlebx,1;EBX的值乘以2orbl,al;BL和AL相加looprdbd2;循环键入字符movtemp,ebx;二进制结果存放TEMPpopecxpopebxpopeaxret第5章模块化程序设计28〔例5-5〕二进制输入程序-4rderr:pushdsmovax,csmovds,axleadx,errmsg;显示错误信息movah,9int21hpopdsjmprdbd1errmsgdb13,10,'Inputerror,enteragain:$'readbdendp第5章模块化程序设计29〔例5-6〕有符号十进制数输入程序-1•十进制有符号整数转换为补码的算法如下:1)判断输入了正数还是负数,并用一个寄存器记录2)输入0~9数字(ASCII码),减30H转换为二进制数3)将前面输入的数值乘10,与刚输入的数字相加得到新的数值4)重复2)、3)步,直到输入一个非数字字符结束5)如果是负数进行求补,转换成补码;否则直接将数值保存第5章模块化程序设计30〔例5-6〕有符号十进制数输入程序-2;数据段count=5arrayddcountdup(0)tempdd?;代码段,主程序movecx,countmovebx,offsetarrayagain:callreadsid;调用子程序,输入一个数据moveax,temp;获得出口参数mov[ebx],eax;存放到数据缓冲区addebx,4calldispcrlfloopagain共享变量传递参数第5章模块化程序设计31〔例5-6〕有符号十进制数输入程序-3;输入有符号十进制数的子程序readsidproc;TEMP=补码表示的二进制数值pusheax;说明:负数用“-”引导pushebxpushecxxorebx,ebx;EBX保存结果xorecx,ecx;ECX为正负标志:0为正,-1为负rsid0:movah,1int21h;输入一个字符cmpal,'+';是“+”,继续jzrsid1第5章模块化程序设计32〔例5-6〕有符号十进制数输入程序-4cmpal,'-';是“-”,设置-1标志jnzrsid2movecx,-1rsid1:movah,1;继续输入字符int21hrsid2:cmpal,'0';不是0~9之间数码,结束jbrsid3cmpal,'9'ja
本文标题:汇编教程05
链接地址:https://www.777doc.com/doc-3717607 .html