函 数
1. 函数的定义
函数定义的一般格式如下:
函数类型 函数名(形式参数表) [reentrant][interrupt m][using n]
形式参数说明
{
局部变量定义
函数体
}
前面部件称为函数的首部,后面称为函数的尾部,格式说明:
1).函数类型
函数类型说明了函数返回值的类型。
2).函数名
函数名是用户为自定义函数取的名字以便调用函数时使用。
3).形式参数表
形式参数表用于列录在主调函数与被调用函数之间进行数据传递的形式参数。
【例】定义一个返回两个整数的最大值的函数max()。
int max(int x,int y)
{ int z;
z=x>y?x:y;
return(z);
}
也可以用成这样:
int max(x,y)
int x,y;
{ int z;
z=x>y?x:y;
return(z);
}
4).reentrant修饰符
这个修饰符用于把函数定义为可重入函数。所谓可重入函数就是允许被递归调用的函数。函数的递归调用是指当一个函数正被调用尚未返回时,又直接或间接调用函数本身。一般的函数不能做到这样,只有重入函数才允许递归调用。
关于重入函数,注意以下几点:
(1)用reentrant修饰的重入函数被调用时,实参表内不允许使用bit类型的参数。函数体内也不允许存在任何关于位变量的操作,更不能返回bit类型的值。
(2)编译时,系统为重入函数在内部或外部存储器中建立一个模拟堆栈区,称为重入栈。重入函数的局部变量及参数被放在重入栈中,使重入函数可以实现递归调用。
(3)在参数的传递上,实际参数可以传递给间接调用的重入函数。无重入属性的间接调用函数不能包含调用参数,但是可以使用定义的全局变量来进行参数传递。
5).interrupt m修饰符
interrupt m是C51函数中非常重要的一个修饰符,这是因为中断函数必须通过它进行修饰。在C51程序设计中,当函数定义时用了interrupt m修饰符,系统编译时把对应函数转化为中断函数,自动加上程序头段和尾段,并按51系统中断的处理方式自动把它安排在程序存储器中的相应位置。
在该修饰符中,m的取值为0~31,对应的中断情况如下:
0——外部中断0
1——定时/计数器T0
2——外部中断1
3——定时/计数器T1
4——串行口中断
5——定时/计数器T2
其它值预留。
编写51中断函数注意如下:
(1)中断函数不能进行参数传递,如果中断函数中包含任何参数声明都将导致编译出错。
(2)中断函数没有返回值,如果企图定义一个返回值将得不到正确的结果,建议在定义中断函数时将其定义为void类型,以明确说明没有返回值。
(3)在任何情况下都不能直接调用中断函数,否则会产生编译错误。因为中断函数的返回是由8051单片机的RETI指令完成的,RETI指令影响8051单片机的硬件中断系统。如果在没有实际中断情况下直接调用中断函数,RETI指令的操作结果会产生一个致命的错误。
(4)如果在中断函数中调用了其它函数,则被调用函数所使用的寄存器必须与中断函数相同。否则会产生不正确的结果。
(5)C51编译器对中断函数编译时会自动在程序开始和结束处加上相应的内容,具体如下:在程序开始处对ACC、B、DPH、DPL和PSW入栈,结束时出栈。中断函数未加using n修饰符的,开始时还要将R0~R1入栈,结束时出栈。如中断函数加using n修饰符,则在开始将PSW入栈后还要修改PSW中的工作寄存器组选择位。
(6)C51编译器从绝对地址8m+3处产生一个中断向量,其中m为中断号,也即interrupt后面的数字。该向量包含一个到中断函数入口地址的绝对跳转。
(7)中断函数最好写在文件的尾部,并且禁止使用extern存储类型说明。防止其它程序调用。
【例】编写一个用于统计外中断0的中断次数的中断服务程序
extern int x;
void int0() interrupt 0 using 1
{
x++;
}
6).using n修饰符
修饰符using n用于指定本函数内部使用的工作寄存器组,其中n的取值为0~3,表示寄存器组号。
对于using n修饰符的使用,注意以下几点:
(1)加入using n后,C51在编译时自动的在函数的开始处和结束处加入以下指令。
{
PUSH PSW ;标志寄存器入栈
MOV PSW,#与寄存器组号相关的常量
……
POP PSW ;标志寄存器出栈
}
(2)using n修饰符不能用于有返回值的函数,因为C51函数的返回值是放在寄存器中的。如寄存器组改变了,返回值就会出错。
2. 函数的调用与声明
一.函数的调用
函数调用的一般形式如下:
函数名(实参列表);
对于有参数的函数调用,若实参列表包含多个实参,则各个实参之间用逗号隔开。
按照函数调用在主调函数中出现的位置,函数调用方式有以下三种:
(1)函数语句。把被调用函数作为主调用函数的一个语句。
(2)函数表达式。函数被放在一个表达式中,以一个运算对象的方式出现。这时的被调用函数要求带有返回语句,以返回一个明确的数值参加表达式的运算。
(3)函数参数。被调用函数作为另一个函数的参数。
二.自定义函数的声明
在C51中,函数原型一般形式如下:
[extern] 函数类型 函数名(形式参数表);
函数的声明是把函数的名字、函数类型以及形参的类型、个数和顺序通知编译系统,以便调用函数时系统进行对照检查。函数的声明后面要加分号。
如果声明的函数在文件内部,则声明时不用extern,如果声明的函数不在文件内部,而在另一个文件中,声明时须带extern,指明使用的函数在另一个文件中。
【例】函数的使用
#include
#include
int max(int x,int y); //对max函数进行声明
void main(void) //主函数
{
int a,b;
SCON=0x52; //串口初始化
TMOD=0x20;
TH1=0XF3;
TR1=1;
scanf(“please input a,b:%d,%d”,&a,&b);
printf(“\n”);
printf(“max is:%d\n”,max(a,b));
while(1);
}
int max(int x,int y)
{ int z;
z=(x>=y?x:y);
return(z);
}
【例24】 外部函数的使用
程序serial_initial.c
#include
#include
void serial_initial(void) //主函数
{
SCON=0x52; //串口初始化
TMOD=0x20;
TH1=0XF3;
TR1=1;
}
程序y1.c
#include
#include
extern serial_initial();
void main(void)
{
int a,b;
serial_initial();
scanf(“please input a,b:%d,%d”,&a,&b);
printf(“\n”);
printf(“max is:%d\n”,a>=b?a:b);
while(1);
}
3. 函数的嵌套与递归
一.函数的嵌套
在一个函数的调用过程中调用另一个函数。C51编译器通常依靠堆栈来进行参数传递,堆栈设在片内RAM中,而片内RAM的空间有限,因而嵌套的深度比较有限,一般在几层以内。如果层数过多,就会导致堆栈空间不够而出错。
【例】 函数的嵌套调用
#include
#include
extern serial_initial();
int max(int a,int b)
{
int z;
z=a>=b?a:b;
return(z);
}
int add(int c,int d,int e,int f)
{
int result;
result=max(c,d)+max(e,f); //调用函数max
return(result);
}
main()
{
int final;
serial_initial();
final=add(7,5,2,8);
printf(“%d”,final);
while(1);
}
二.函数的递归
递归调用是嵌套调用的一个特殊情况。如果在调用一个函数过程中又出现了直接或间接调用该函数本身,则称为函数的递归调用。
在函数的递归调用中要避免出现无终止的自身调用,应通过条件控制结束递归调用,使得递归的次数有限。
下面是一个利用递归调用求n!的例子。
【例】递归求数的阶乘n!。
在数学计算中,一个数n的阶乘等于该数本身乘以数n-1的阶乘,即n!=n´(n-1)!,用n-1的阶乘来表示n的阶乘就是一种递归表示方法。在程序设计中通过函数递归调用来实现。
程序如下:
#include
#include
extern serial_initial();
int fac(int n) reentrant
{
int result;
if (n= =0)
result=1;
else
result=n*fac(n-1);
return(result);
}
main()
{
int fac_result;
serial_initial();
fac_result=fac(11);
printf(“%d\n”,fac_result);
}