汇编语言中‘子程序’的设计方法

汇编语言是各种计算机语言中与硬件关系最为密切、最直接的语言,是时空效率最高的语言,它能够利用计算机所有硬件特性并能直接控制硬件,所以在计算机应用系统设计和过程控制中是必不可少的.目前教学中采用8086/8088汇编语言系统组织教学仍是最佳选择.其中子程序技术是一种解决重复性问题的重要设计方法,采用子程序结构可以简化源程序书写、提高程序存储效率、减少出错率、增加程序的易读性和可维护性,并且有利用子程序资源的组织和使用.设计子程序时,除了必需要考虑的程序调用、返回和完成特定功能的指令序列外,还必须注意解决子程序设计中带有的共性的一些问题,即:现场保护、参数传递、子程序的嵌套与递归调用、编写子程序说明文档等.

1 现场保护 

现场保护的目的是调用子程序之后,能够返回主程序继续执行.因此要对子程序中用到的寄存器,堆栈进行必要的保护.

1 1 寄存器保护因为汇编语言程序中的主要操作对象是CPU中的各寄存器,对那些主程序和子程序中都会用到的一些寄存器要在子程序使用之前进行保护.寄存器保护最好是在子程序中进行,并且在子程序中进行恢复,这样子程序显得更完整.其方法是使用堆栈,由于指令系统中制定了规范的进栈指令PUSH和出栈指令POP,并会自动修改堆栈指针,只要在程序设计中注意8086/8088的堆栈是否按"后进先出"的原则组织的.

1 2 堆栈保护子程序是利用调用(CALL)指令和返回(RET)指令来实现正确的调用和返回的.因为CALL命令执行时压入堆栈的断点地址就是供子程序返回主程序时的地址,编程时一定要注意子程序的类型属性,即是段内调用还是段间调用.段内调用和返回为NEAR属性,段间调 王艳玲,等谈谈汇编语言中子程序的设计方法37用和返回为FAR属性.8086/8088的汇编程序用子程序定义PROC的类型属性来确定CALL和RET指令的属性.如果所定义的子程序是FAR属性,那么对它的调用和返回一定都是FAR属性;如果所定义的子程序是NEAR属性,那么对它的调用和返回也一定是NEAR属性.这样用户只是在定义子程序时考虑它的属性,而CALL和RET指令的属性就可以由汇编程序来确定了.另外,进入子程序后再使用堆栈时也必须保证压入和弹出字节数一致,如果在这里堆栈存取出错,必然会导致返回地址的错误.

2 参数传递 

主程序在调用子程序时,经常要向子程序传递一些参数或控制信息,子程序执行后,也常需要把运行的结果返回调用程序.这种信息传递称为参数传递,其常用的方法有寄存器传递、内存固定单元传递、堆栈传递.

2 1 寄存器传递由主程序将要传递的参数装入事先约定的寄存器中,转入子程序后再取出进行处理,这种方法受CPU内部寄存器数量限制,因此只适于传递少量参数的场合,如一些常见的软件延时子程序,均是利用某寄存器传递循环计数器初值.

2 2 通过内存固定单元的传递此方法适于大量传递参数时使用,它是在内存中开辟特定的一片区域用于传递参数.主程序和子程序都按事先约定在指定的存储单元中进行数据交换,这种方法要占用一定数量的存储单元.不足之处是信息易被修改,不利于模块化设计.

2 3 通过堆栈实现参数传递这种方法是先在主程序中把参数和参数地址压入堆栈,在子程序中取出使用,由于堆栈操作不占用寄存器,并且堆栈单元使用后可自动释放,反复使用,便于实现数据隔离和模块化设计.使用这种方法时,当子程序返回后,这些参数就不在有用了,应当丢弃.这时可以利用带立即数的返回指令修改指针,使其指向参数入栈以前的值.

3 子程序嵌套与递归调用  

汇编语言中子程序的嵌套只要堆栈空间允许,一般不受嵌套层次限制.嵌套子程序设计中,应注意寄存器的保护和恢复,避免各层子程序之间寄存器冲突.递归子程序的设计必须保证每次调用都不破坏以前调用时所用的参数和中间结果.为保证每次调用的正确,一般把每次调用的参数、有关寄存器的内容以及中间结果进栈保存.

4 子程序说明文档  

一般来说子程序是要反复使用或提供用户使用,所以编写子程序时应尽量采用较好的算法,使子程序运行速度比较快,又节省内存.同时还应最大限度地满足今后程序维护与使用的需要.在设计子程序的同时就应当建立相应的说明文档,清楚地描述子程序的功能和调用方法.通常子程序说明文档应包括:子程序名称、子程序功能、入口参数、出口参数、工作寄存器、工作单元及最后修改日期等

时间: 2024-10-27 12:09:40

汇编语言中‘子程序’的设计方法的相关文章

java-Java语言中的get set方法有什么用

问题描述 Java语言中的get set方法有什么用 Java语言中的get set方法有什么用,直接调用字段不是更简单么?有人说验证输入,但是也没有验证的代码写出来啊? 解决方案 便于加入验证逻辑.也许你说,现在没有验证逻辑.但是如果直接访问字段,以后再加上验证逻辑,那么调用代码就要修改了,不是麻烦么 解决方案二: 一个类要把自己的属性封装起来,不能让其他类直接访问.通过get和set方法的设置可以控制访问权限.不过一般的类都是get和set同时开放,在这种情况下和直接访问属性是一样效果.再如

汇编语言中“[]”的用法

"[]"的用法在"常见问题"已经有所说明,引用如下: 1.push dword ptr [024c1100] 压栈024c1100值的双字 2.cmp eax,[ebp+14] eax-ebp+14的有效值,不保留值,主要看标志位 3.cmp byte ptr [eax],46 字节型eax-46,看标志位 4.lea eax,[edx-02] 把edx-02的有效值(一个地址值)给eax 5.mov ecx,[edx+08] edx+8值作为地址,此地址所指向的值

C语言中assert的用法

以下是对C语言中assert的使用方法进行了介绍,需要的朋友可以参考下   assert宏的原型定义在<assert.h>中,其作用是如果它的条件返回错误,则终止程序执行,原型定义:#include <assert.h> void assert( int expression );assert的作用是现计算表达式 expression ,如果其值为假(即为0),那么它先向stderr打印一条出错信息, 然后通过调用 abort 来终止程序运行.请看下面的程序清单badptr.c:

java语言中new A( ).B( )意思是不是A类的一个对象调用B方法

问题描述 java语言中new A( ).B( )意思是不是A类的一个对象调用B方法 java语言中new A( ).B( )意思是不是A类的一个对象调用B方法如果是new A( ).B( ).......N( )呢这样符合java语法规范吗 解决方案 调用A对象的B方法,因为如果A类中的B方法不是静态方法,那么外面想要调用的话就必须要先将A实例化,也就是是 new A(),之后调用它的B方法,也就是new A( ).B( ).这个是对象匿名的用法,是符合规范的.使用这种方法的话,就不需要将B方

visual basic-改进Visual Basic语言中的Round实现自定义的方法

问题描述 改进Visual Basic语言中的Round实现自定义的方法 VB6.0 语言实现一舍二入请问怎么实现呢?Round可以自定义么 解决方案 自己写一个,比如保留2位: x = 1.2345 x = Int(x * 100 + 8) / 100

c语言课程设计-c语言中a=b--&amp;amp;lt;=a||a b!=c怎么算

问题描述 c语言中a=b--<=a||a b!=c怎么算 c语言中a=b--<=a||a b!=c怎么算a=1,b=2,c=3,求计算过程以及计算后ab的值 解决方案 先找逻辑运算符"||" 左边:(a=b--) <= a 计算b--的值,b--的值为2(得到这个值也就让b的值变成了1),赋值给a后a的值为2,(a=b--)的值也就是a的值为2,2<=2成立,所以左边为真 右边:(a+b)!=c,a+b的值为3,c的值为3,所以右边为假 真||假的结果为真 b-

简述C语言中system()函数与vfork()函数的使用方法_C 语言

C语言system()函数:执行shell命令头文件: #include <stdlib.h> 定义函数: int system(const char * string); 函数说明:system()会调用fork()产生子进程, 由子进程来调用/bin/sh-c string 来执行参数string 字符串所代表的命令, 此命令执行完后随即返回原调用的进程. 在调用system()期间SIGCHLD 信号会被暂时搁置,SIGINT 和SIGQUIT 信号则会被忽略. 返回值: 1.如果 sy

详解C语言中fseek函数和ftell函数的使用方法_C 语言

fseek函数: int fseek(FILE * _File, long _Offset, int _Origin); 函数设置文件指针stream的位置.如果执行成功,stream将指向以fromwhere为基准,偏移offset(指针偏移量)个字节的位置,函数返回0.如果执行失败则不改变stream指向的位置,函数返回一个非0值. 超出文件末尾位置,还是返回0.往回偏移超出首位置,还是返回0,小心使用. 第一个参数stream为文件指针. 第二个参数offset为偏移量,正数表示正向偏移,

Java 语言中 Enum 类型的使用介绍

Enum 类型的介绍 枚举类型(Enumerated Type) 很早就出现在编程语言中,它被用来将一组类似 的值包含到一种类型当中.而这种枚举类型的名称则会被定义成独一无二的类型描述符,在这一点上和常量的 定义相似.不过相比较常量类型,枚举类型可以为申明的变量提供更大的取值范围. 举个例子来说明 一下,如果希望为彩虹描绘出七种颜色,你可以在 Java 程序中通过常量定义方式来实现. 清单 1. 常量定义 Public static class RainbowColor { // 红橙黄绿青蓝紫