首页 > 代码库 > C++学习笔记之函数指针

C++学习笔记之函数指针

与数据项类似,函数也有地址。函数的地址是存储其机器语言代码的内存开始的地方。

一、函数指针的基础知识

假设要设计一个名为estimate()的函数,估算编写指定行数代码所需时间,并且希望不同的程序员都使用该函数,并且该函数允许每个程序员提供自己的算法来估计时间。为实现这种目标,采用的机制是,将程序员要使用的算法函数地址传给estimate(),必须完成以下工作:

  • 获取函数地址
  • 声明一个函数指针
  • 用函数指针来调用函数

1.获取函数地址

使用函数名(后面不跟参数)即可。如:think()是一个函数,则think就是该函数的地址。要将函数作为参数进行传递,必须传递函数名。

一定要注意区分传递的是函数的地址还是函数的返回值:

1 process(think);    //传递函数think()的地址给process()2 thought(think()); //传递函数think()的返回值给though()

 

2.声明函数指针

声明应指定指针所指向的函数的返回类型以及函数的特征标(参数列表),假设Pam编写了一个估算时间的函数,其原型如下:

1 double pam(int);    // 函数声明

 

则正确的指针类型声明如下:

1 double (*pf) (int); 2 // pf指向一个输入一个int参数,返回一个double值得函数

 

可以看出,就是将函数声明中的函数名替换为*pf, 于是pf就成为指向函数的指针,由指针的基本知识可以知道,(*pf)也是函数。

必须在声明中用括号将*pf括起,否则如下:

1 double *pf(int);

 

这相当于声明了一个函数pf,它的返回值是一个指向double的指针,与我们想要声明的函数指针完全不是一码事。

正确声明pf之后,便可以将相应的函数地址赋给它:

1 double pam(int);2 double (*pf)(int);3 pf = pam;

 

现在指针pf就指向函数pam()了。再次强调pam()的特征标和返回类型必须与pf相同。

假设现在要将编写的代码行数和估算算法(如pam()函数)的地址传递给estimate(),原型如下:

1 void estimate(int lines, double (*pf)(int));

 

要让estimate()使用pam()函数,需要将pam()的地址传给它:

1 estimate(50, pam);

 

3.使用指针来调用函数

上面讲过,(*pf)扮演的角色与函数名相同,因此使用(*pf)时,只需将它看作函数名即可:

1 double pam(int);2 double (*pf)(int);3 pf = pam;4 double x = pam(4);   // 用函数名调用函数5 double y = (*pf)(5);  // 用指针pf调用函数

 

但是,C++也允许像使用函数名那样使用pf:

double y = pf(5);  //与double y = (*pf)(5)效果一样

 

为何pf与(*pf)等价呢?因为存在两种看法,一种学派认为,由于pf是函数指针,而*pf是函数,因此应将(*pf)()用作函数调用;而另一种学派认为,由于函数名是指向该函数的指针,指向函数指针的行为应与函数名相似,因此应将pf()用作函数调用。看上去都是合理的说法,因此C++进行了折衷--两种方法都正确,虽然在逻辑上相互冲突。

二、函数指针示例

 1 /*函数指针示例*/ 2 #include <iostream> 3  4 using namespace std; 5  6 //将会被调用的两个函数的声明 7 double betsy(int); 8 double pam(int); 9 10 //estimate()声明11 void estimate(int lines, double (*pf)(int));12 13 int main()14 {15     int code;16 17     cout << "请输入代码行数:";18     cin >> code;19     cout << "Betsy的评估结果:\n";20     estimate(code, betsy);21     cout << "Pam的评估结果:\n";22     estimate(code, pam);23     return 0;24 }25 26 double betsy(int lns)27 {28     return 0.05 * lns;29 }30 31 double pam(int lns)32 {33     return 0.03 * lns + 0.0004 * lns * lns;34 }35 36 void estimate(int lines, double (*pf)(int))37 {38     cout << lines << "行代码将耗时 ";39     cout << (*pf)(lines) << " 小时\n";40 }

 

运行结果: