函数指针
与数据项类似,函数也有地址。函数的地址是存储其机器语言代码的内存的开始地址。通常,这样地址对用户而言是透明的,但对操作系统而言却是很有用的。例如,可以编写将另一个函数的地址作为参数的函数。这样第一个函数就能够找到第二个函数并执行所需的操作。函数地址作为参数传递,实现了在不同时间使用不能函数,可以像对待普通变量一样使用函数。
首先通过一个例子来阐述这一过程。假设要设计一个名为estimate()的函数,估算编写指定行数的代码所需的时间,并且希望不同的程序员都将使用该函数。对于所有的用户来说,estimate()中的一部分代码是相同的,但该函数允许每个程序员提供自己的算法来估算时间。为实现这种目标,采用的机制是,将程序员要使用的算法函数的地址传给estimate()。为此,必须能够完成下面的工作:
l 获取函数的地址
l 声明一个函数指针
l 使用函数指针来调用函数。
1.获取函数的地址
获取函数的地址很简单:只要使用函数名(不带参数)即可。也就是说,如果think()是一个函数,那么think就是该函数的地址。这里需要区分一点,就是传递给调用函数的是函数的返回值还是函数的首地址:
Process(think );
Process(think () );
对第一个Process()调用,传递的是函数的首地址;第二个则传递的是函数的返回值。
2.声明函数指针
由指针的性质,声明一个指向函数的指针时,也必须指定指针指向的是函数类型,这意味着声明应指定函数的返回值类型、参数列表等。例如有如下声明:
doublepam ( int );
普通函数声明
double(*pf)( int );
指针类型函数声明
这与pam()声明类似,只是将pam替换成为*(pf)。由于pam是函数,因此(*pf)也是函数。而如果(*pf)是函数,那么pf就是指向函数的指针了。
这里还要看到,(*pf)是有括号的,因为某种优先级的原因,*( pf )( int )和*pf( int )是完全不同的两个概念。前者的pf是一个指向函数的指针,而后者的pf()是一个返回指针的函数。
3.使用指针来调用函数
前面讲过,(*pf)扮演的角色与函数名相同,因此只需将原本是函数的地方换成(*pf)就好了:
doublepam ( int );
double(*pf) ( int );
pf= pam; //此时pf指向函数pam;
double x = pam (4);
double y= (*pf) (5); //使用指针调用函数;
实例剖析:
下面的代码演示了如何使用函数指针。它两次调用estimate()函数,一次传递betsy()函数的地址,另一次则传递pam()函数的地址。在第一种情况下,estimate()使用besty()函数计算所需的小时数;第二种情况下,使用pam()进行计算。这种设计模式值得我们深究。
#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
double besty(int );
double pam(int);
void estimate(int lines,double(*pf)(int));
int main()
{
int code;
cout<<" How many lines of code do you need? ";
cin>>code;
cout<<"Here is besty's estimate:\n";
estimate(code,besty);
cout<<"Here is pam's estimate:\n";
estimate(code,pam);
return 0;
}
double besty(int lns)
{
return 0.05*lns;
}
double pam(int lns)
{
return 0.03*lns+0.0004*lns*lns;
}
void estimate(int lines,double(*pf)(int))
{
cout<<lines<<" lines will take ";
cout<< (*pf)(lines)<<" hours"<<endl;
}
下面是2次运行该程序的结果: