与数据项相似,函数也有地址。函数的地址就是存储其机器语言代码的内存的开始地址
函数指针的基础知识
1.获取函数的地址
一定要区分传递的是函数的地址还是函数的返回值:
如果think()是一个函数则:
process(think);//传递地址
thought(think());//传递函数的返回值
process()调用使得process()函数能够在其内部调用think()函数。thought()调用首先调用think()函数,然后将think()的返回值传递给thought()函数
2.声明函数指针
声明指向某种数据类型的指针时,必须指定指针指向的类型。同样,声明指向函数的指针时,也必须指定指针指向的函数类型。
如:
//某个函数声明:
double pam(int);
//指针类型声明:
double (*pf)(int);
与pam()声明类似,这是将pam替换成了(*pf)。pam是函数,所以(*pf)也是函数,pf就是函数指针。
由于()的优先级比*高,所以不能省略
double (*pf)(int);//pf是一个指向函数的指针
double *pf(int);//pf是一个返回指针的函数
正确地声明pf之后,便可以将相应函数的地址赋给它,注意,pam()的特征标(参数类型和个数)和返回类型必须与pf相同。
double pam(int);
double (*pf)(int);
pf = pam;
现在我们写一个estimate()函数来估算编写指定行数代码所需的时间:
void estimate(int lines,double (*pf)(int));//原型
estimate(50,pam);//传递pam()的地址
3.使用指针来调用函数
(*pf)与函数名相同,因此使用(*pf)时可以将它当作函数名:
double pam(int);
double (*pf)(int);
pf = pam;
double x = pam(4);
double y = (*pf)(5);
实际上C++也允许像使用函数名一样使用pf
double y = pf(5);
关于为何(*pf)与pf等价:
一种学派认为,由于pf是函数指针,而 *pf是函数,因此可以将(*pf)()用作函数调用。
另一种学派认为,由于函数名是指向该函数的指针,指向函数的指针行为应与函数名相似,因此应将pf()用作函数调用使用。
下面以一道具体的实例来演示:
// fun_ptr.cpp -- pointers to functions
#include <iostream>