- 0
- 0
- 约2.58万字
- 约 166页
- 2022-05-24 发布于重庆
- 举报
主程序: 地址 机器码 指令 0005 12 ** LCALL DELAY ;第一次调用子程序 0008 ****** MOV P1,#0FFH ;LCALL指令的下 一条指令首址0008H称为断点地址 第八十六页,共一百六十六页。 子程序: 0013 ****** MOV R3,#0FFH ;子程序开始 001C 22 RET ;子程序返回 主程序两次调用子程序及子程序返回过程如图所示。 ?? 第八十七页,共一百六十六页。 子程序两次调用、返回过程示意图 第八十八页,共一百六十六页。 子程序只需书写一次,主程序可以反复调用它。CPU执行LCALL指令所进行的具体操作(以第一次调用为例)是: (a) PC的自动加1功能使PC=0008H,指向下一条指令MOV P1,#0FFH的首址,PC中即为断点地址; (b) 保存PC中的断点地址0008H; (c) 将子程序DELAY的入口地址0013H赋给PC,PC=0012H; (d) 程序转向DELAY子程序运行。 第八十九页,共一百六十六页。 CPU执行RET指令的具体操作(以第一次调用为例)是: (a) 取出执行调用指令时保存的断点地址0008H,并将它赋给PC,PC=0008H; (b) 程序转向断点处继续执行主程序。 从以上分析来看,在子程序调用过程中,断点地址0008H是自动保存和取出的,那么断点地址究竟存放在什么地方呢?这里引出一个新的存储区域概念——堆栈,它是一个存放临时数据(例如断点地址)的内存区域。堆栈的巧妙设计使程序员不必操心数据的具体存放地址。 ② 子程序嵌套。 第九十页,共一百六十六页。 修改上面的程序,将一个灯的闪烁过程也编成子程序形式。修改后的源程序如下: ORG 0000H MAIN: MOV A,#0FEH ;送显示初值 COUN: ACALL FLASH ;调闪烁子程序 RL A ;A左移,下一个灯闪烁 SJMP COUN ;循环不止 第九十一页,共一百六十六页。 FLASH: MOV R0,#10 ;送闪烁次数 FLASH1: MOV P1,A ;点亮LED LCALL DELAY ;延时 MOV P1,#0FFH ;熄灭灯 LCALL DELAY ;延时 DJNZ R0,FLASH1 ;闪烁次数不够10次,继续 RET 第九十二页,共一百六十六页。 DELAY: MOV R3,#0FFH ;延时子程序 DEL2: MOV R4,#0FFH DEL1: NOP DJNZ R4,DEL1 DJNZ R3,DEL2 RET END 第九十三页,共一百六十六页。 上面程序中,主程序调用了闪烁子程序FLASH,闪烁子程序中又调用延时子程序DELAY,这种主程序调用子程序,子程序又调用另外的子程序的程序结构,称为子程序的嵌套。一般来说,子程序嵌套层数理论上是无限的,但实际上,受堆栈深度的影响,嵌套层数是有限的。 与子程序的多次调用不同,嵌套子程序的调用过程如图所示。 第九十四页,共一百六十六页。 例4.10程序中嵌套子程序的执行过程 第九十五页,共一百六十六页。 例4.11 查表子程序。假设a、b均小于10,计算c=a2+b2,其中a事先存在内部RAM的31H单元,b事先存在32H单元,请把c存入33H单元。 (1) 题意分析。 本例两次使用平方的计算,在前面的例4.9中已经编过查平方表得到平方值的程序,在此我们采用把求平方编为子程序的方法。 (2) 汇编语言源程序。 第九十六页,共一百六十六页。 ORG 0000H ;主程序 MOV SP,#3FH ;设置栈底 MOV A,31H ;取数a存放到累加器A中 作
原创力文档

文档评论(0)