3.6子程序设计把功能相对独立的程序段单独编写和调试,作为一个相对独立的模块供程序使用,就形成子程序子程序可以实现源程序的模块化,可简化源程序结构,可以提高编程效率主程序(调用程序)需要利用CALL指令调用子程序(被调用程序)子程序需要利用RET指令返回主程序3.6.1过程定义和子程序编写汇编语言中,子程序要用一对过程伪指令PROC和ENDP声明,格式如下:过程名PROC[NEAR|FAR]……;过程体过程名ENDP可选的参数指定过程的调用属性。没有指定过程属性,则采用默认属性NEAR属性(段内近调用)的过程只能被相同代码段的其他程序调用FAR属性(段间远调用)的过程可以被相同或不同代码段的程序调用子程序编写注意事项⑴子程序要利用过程定义伪指令声明⑵子程序最后利用RET指令返回主程序,主程序执行CALL指令调用子程序⑶子程序中对堆栈的压入和弹出操作要成对使用,保持堆栈的平衡⑷子程序开始应该保护使用到的寄存器内容,子程序返回前相应进行恢复⑸子程序应安排在代码段的主程序之外,最好放在主程序执行终止后的位置(返回DOS后、汇编结束END伪指令前),也可以放在主程序开始执行之前的位置例3.15用显示器功能调用输出一个字符的子程序;主程序moval,‘?’;主程序提供显示字符calldpchar;调用子程序;子程序:显示AL中的字符dpcharproc;过程定义,过程名为dpcharpushax;顺序入栈,保护寄存器pushbxmovbx,0movah,0eh;显示器0EH号输出一个字符功能int10hpopbx;逆序出栈,恢复寄存器popaxret;子程序返回dpcharendp;过程结束例3.15源程序;wj0315.asm.modelsmall.stack.codestart:movax,@datamovds,axmoval,‘?’;主程序提供显示字符calldpchar;调用子程序movax,4c00hint21h主程序部分本程序不需要数据段例3.15源程序(续)dpcharproc;过程定义,过程名为dpcharpushax;顺序入栈,保护寄存器pushbxmovbx,0movah,0eh;显示器0EH号输出一个字符功能int10hpopbx;逆序出栈,恢复寄存器popaxret;子程序返回dpcharendp;过程结束endstart子程序安排在主程序执行终止后的位置子程序编写注意事项(续)⑹子程序允许嵌套和递归⑺子程序可以与主程序共用一个数据段,也可以使用不同的数据段(注意修改DS),还可以在子程序最后设置数据区(利用CS寻址)⑻子程序的编写可以很灵活,例如具有多个出口(多个RET指令)和入口,但一定要保证堆栈操作的正确性⑼处理好子程序与主程序间的参数传递问题⑽提供必要的子程序说明信息例3.16显示以“0”结尾字符串的嵌套子程序;数据段msgdb'Well,Imadeit!',0;代码段(主程序)movsi,offsetmsg;主程序提供显示字符串calldpstri;调用子程序例3.16子程序;子程序dpstri:显示DS:SI指向的字符串(以0结尾)dpstriprocpushaxdps1:moval,[si];取显示字符incsicmpal,0;是结尾,则显示结束jzdps2calldpchar;调用字符显示子程序jmpdps1dps2:popaxretdpstriendp;子程序dpchar:显示AL中的字符(同例题3.15)含数据区的子程序;子程序HTOASC:十六进制数转换为ASCII码HTOASCprocpushbxmovbx,offsetASCIIandal,0fhxlatCS:ASCII;换码:AL←CS:[BX+AL]popbxret;数据区ASCIIdb30h,31h,32h,33h,34h,35h,36h,37h,38h,39hdb41h,42h,43h,44h,45h,46hHTOASCendp多出口子程序;子程序HTOASC:十六进制数转换为ASCII码HTOASCprocandal,0fhcmpal,9jbehtoasc1addal,37h;是A~F,加37Hret;子程序返回htoasc1:add,30h;是0~9,加30Hret;子程序返回HTOASCendp参数传递主程序与子程序间一个主要问题是参数传递入口参数(输入参数):主程序调用子程序时,提供给子程序的参数出口参数(输出参数):子程序执行结束返回给主程序的参数参数的具体内容传数值:传送数据本身传地址:传送数据的主存地址常用的参数传递方法寄存器共享变量堆栈3.6.2用寄存器传递参数最简单和常用的参数传递方法是通过寄存器,只要把参数存于约定的寄存器中就可以了由于通用寄存器个数有限,这种方法对少量数据可以直接传递数值,而对大量数据只能传递地址采用寄存器传递参数,注意带有出口参数的寄存器不能保护和恢复,带有入口参数的寄存器可以保护、也可以不保护,但最好能够保持一致dpchardpstriHTOASC例3.17用寄存器传递参数显示字符串;数据段msgdb'Well,Imadeit!',0;代码段(主程序)movsi,offsetmsg;SI寄存器传递参数:字符串地址calldpstri;调用子程序例3.17用寄存器传递参数显示字符串(续);代码段(子程序)dpstriproc;显示以0结尾的字符处pushax;入口参数:SI=字符串地址pushdxdps1:movdl,[si];通过SI使用参数cmpdl,0jzdps2movah,2int21hincsijmpdps1dps2:popdxpopaxretdpstriendp例3.18从键盘输入有符号十进制数;数据段count=10arraydwcountdup(0);代码段(主程序)movcx,countmovbx,offsetarrayagain:callread;调用子程序,输入一个数据mov[bx],ax;存放出口参数incbxincbxcalldpcrlf;调用子程序:光标回车loopagain将ASCII码转换为二进制数的算法①首先判断输入正数还是负数,并用一个寄存器记录下来;②接着输入0~9数字(ASCII码),并减30H转换为二进制数;③然后将前面输入的数值乘10,并与刚输入的数字相加得到新的数值;④重复②、③步,直到输入一个非数字字符结束;⑤如果是负数进行求补,转换成补码;否则直接将数值保存。例3.18从键盘输入有符号十进制数(续1)readproc;输入有符号十进制数pushbx;出口参数:AXpushcx;说明:负数用“-”引导pushdxxorbx,bx;BX保存结果xorcx,cx;CX为正负标志,0为正,-1为负movah,1;输入一个字符int21hcmpal,'+';是“+”,继续输入字符jzread1cmpal,'-';是“-”,设置-1标志jnzread2movcx,-1例3.18从键盘输入有符号十进制数(续2)read1:movah,1;继续输入字符int21hread2:cmpal,‘0’;不是0~9之间的字符,输入结束jbread3cmpal,'9'jaread3subal,30h;是0~9之间的字符,转换为二进制数;利用移位指令,实现数值乘10:BX←BX×10shlbx,1movdx,bxshlbx,1shlbx,1addbx,dx例3.18从键盘输入有符号十进制数(续3)movah,0addbx,ax;已输入数值乘10后,与新输入数值相加jmpread1;继续输入字符read3:cmpcx,0;是负数,进行求补jzread4negbxread4:movax,bx;设置出口参数popdxpopcxpopbxret;子程序返回readendp例3.18从键盘输入有符号十进制数(续4)dpcrlfproc;使光标回车换行的子程序pushaxpushdxmovah,2movdl,0dhint21hmovah,2movdl,0ahint21hpopdxpopaxretdpcrlfendp3.6.3用共享变量传递参数子程序和主程序使用同一个变量名存取数据就是利用共享变量(全局变量)进行参数传递如果变量定义和使用不在同一个源程序中,需要利用PUBLIC、EXTREN声明如果主程序还要利用原来的变量值,则需要保护和恢复利用共享变量传递参数,子程序的通用性较差,但特别适合在多个程序段间、尤其在不同的程序模块间传递数据例3.19用共享变量传递参数显示字符串;数据段msgdb'Well,Imadeit!',0tempdw?;**共享变量;代码段(主程序)movsi,offsetmsgmovtemp,si;**共享变量传递参数calldpstri;调用子程序例3.19用共享变量传递参数显示字符串(续);代码段(子程序)dpstriproc;显示以0结尾的字符处pushax;入口参数:temp=字符串地址pushdxmovsi,temp;**通过temp获得参数……;后同例3.16A程序例3.20向显示器输出有符号十进制数;数据段count=10arraydw1234,-1234,0…wtempdw?;代码段(主程序)movcx,countmovbx,offsetarrayagain:movax,[bx]movwtemp,ax;将入口参数存放到共享变量callwrite;调用子程序,显示一个数据incbxincbxcalldpcrlf;光标回车换行loopagain将二进制数转换为ASCII码的算法①首先判断数据是零、正数或负数,是零显示“0”退出;②是负数,显示“-”,求数据的绝对值;③接着数据除以10,余数加30H转换为ASCII码压入堆栈;④重复③步,直到商为0结束;⑤依次从堆栈弹出各位数字,进行显示。例3.20向显示器输出有符号十进制数(续1)writeproc;显示有符号10进制数的通用子程序pushax;入口参数:共享变量wtemppushbxpushdxmovax,wtemp;取出显示数据testax,ax;判断数据是零、正数或负数jnzwrite1movdl,'0';是零,显示“0”后退出movah,2int21hjmpwrite5例3.20向显示器输出有符号十进制数(续2)write1:jnswrite2;是负数,显示“-”movbx,ax;AX数据暂存于BXmovdl,'-'movah,2int21hmovax,bxnegax;数据求补(绝对值)write2:movbx,10pushbx;10压入堆栈,作为退出标志例3.20向显示器输出有符号十进制数(续3)write3:cmpax,0;数据(商)为零,转向显示jzwrite4subdx,dx;扩展被除数DX.AXdivbx;数据除以10:DX.AX÷10adddl,30h;余数(0~9)转换为ASCII码pushdx;数据各位先低位后高位压入堆栈jmpwrite3write4:popdx;数据各位先高位后低位弹出堆栈cmpdl,10;是结束标志10,则退出jewrite5例3.20向显示器输出有符号十进制数(续4)movah,2;进行显示int21hjmpwrite4write5:popdxpopbxpopaxret;子程序返回writeendp3.6.4用堆栈传递参数参数传递还可以通过堆栈这个临时存储区。主程序将入口参数压入堆栈,子程序从堆栈中取出参数;子程序将出口参数压入堆栈,主程序弹出堆栈取得它们采用堆栈传递参数是程式化的,它是编译程序处理参数传递、以及汇编语言与高级语言混合编程时的常规方法例3.21用堆栈传递参数显示字符串;数据段msgdb'Well,Imadeit!',0;代码段(主程序)movsi,offsetmsgpushsi;**入口参数压入堆栈calldpstri;调用子程序addsp,2;**平衡堆栈例3.21用堆栈传递参数显示字符串(续)dpstriproc;显示以0结尾的字符处pushbp;**入口参数:堆栈=字符串地址movbp,sp;**通过BP获得堆栈内